/* * Copyright (C) 2014 Cloudius Systems, Ltd. */ #include "log.hh" #include #include #include #include namespace logging { registry& logger_registry(); const std::map log_level_names = { { log_level::trace, "trace" }, { log_level::debug, "debug" }, { log_level::info, "info" }, { log_level::warn, "warn" }, { log_level::error, "error" }, }; } using namespace logging; std::ostream& operator<<(std::ostream& out, log_level level) { return out << log_level_names.at(level); } std::istream& operator>>(std::istream& in, log_level& level) { sstring s; in >> s; if (!in) { return in; } for (auto&& x : log_level_names) { if (s == x.second) { level = x.first; return in; } } in.setstate(std::ios::failbit); return in; } namespace logging { logger::logger(sstring name) : _name(std::move(name)) { logger_registry().register_logger(this); } logger::logger(logger&& x) : _name(std::move(x._name)), _level(x._level.load(std::memory_order_relaxed)) { logger_registry().moved(&x, this); } logger::~logger() { logger_registry().unregister_logger(this); } void logger::really_do_log(log_level level, const char* fmt, stringer** s, size_t n) { const char* p = fmt; while (*p != '\0') { if (*p == '{' && *(p+1) == '}') { p += 2; if (n > 0) { (*s++)->append(std::cout); --n; } else { std::cout << "???"; } } else { std::cout << *p++; } } std::cout << "\n"; } void registry::set_all_loggers_level(log_level level) { std::lock_guard g(_mutex); for (auto&& l : _loggers | boost::adaptors::map_values) { l->set_level(level); } } log_level registry::get_logger_level(sstring name) const { std::lock_guard g(_mutex); return _loggers.at(name)->level(); } void registry::set_logger_level(sstring name, log_level level) { std::lock_guard g(_mutex); _loggers.at(name)->set_level(level); } std::vector registry::get_all_logger_names() { std::lock_guard g(_mutex); auto ret = _loggers | boost::adaptors::map_keys; return std::vector(ret.begin(), ret.end()); } void registry::register_logger(logger* l) { std::lock_guard g(_mutex); _loggers[l->name()] = l; } void registry::unregister_logger(logger* l) { std::lock_guard g(_mutex); _loggers.erase(l->name()); } void registry::moved(logger* from, logger* to) { std::lock_guard g(_mutex); _loggers[from->name()] = to; } sstring pretty_type_name(const std::type_info& ti) { int status; std::unique_ptr result( abi::__cxa_demangle(ti.name(), 0, 0, &status), std::free); return result.get() ? result.get() : ti.name(); } registry& logger_registry() { static registry g_registry; return g_registry; } } namespace boost { template <> logging::log_level lexical_cast(const std::string& source) { std::istringstream in(source); logging::log_level level; // Using the operator normall fails. if (!::operator>>(in, level)) { throw boost::bad_lexical_cast(); } return level; } } std::ostream& operator<<(std::ostream&out, std::exception_ptr eptr) { if (!eptr) { out << ""; return out; } try { std::rethrow_exception(eptr); } catch(...) { auto tp = abi::__cxa_current_exception_type(); if (tp) { out << logging::pretty_type_name(*tp); } else { // This case shouldn't happen... out << ""; } // Print more information on some familiar exception types try { throw; } catch(const std::system_error &e) { out << " (error " << e.code() << ", " << e.code().message() << ")"; } catch(const std::exception& e) { out << " (" << e.what() << ")"; } } return out; }