1 // 2 // Copyright(c) 2015 Gabi Melman. 3 // Distributed under the MIT License (http://opensource.org/licenses/MIT) 4 // 5 6 #pragma once 7 8 // Loggers registy of unique name->logger pointer 9 // An attempt to create a logger with an already existing name will be ignored 10 // If user requests a non existing logger, nullptr will be returned 11 // This class is thread safe 12 13 #include <spdlog/details/null_mutex.h> 14 #include <spdlog/logger.h> 15 #include <spdlog/async_logger.h> 16 #include <spdlog/common.h> 17 18 #include <chrono> 19 #include <functional> 20 #include <memory> 21 #include <mutex> 22 #include <string> 23 #include <unordered_map> 24 25 namespace spdlog 26 { 27 namespace details 28 { 29 template <class Mutex> class registry_t 30 { 31 public: 32 register_logger(std::shared_ptr<logger> logger)33 void register_logger(std::shared_ptr<logger> logger) 34 { 35 std::lock_guard<Mutex> lock(_mutex); 36 auto logger_name = logger->name(); 37 throw_if_exists(logger_name); 38 _loggers[logger_name] = logger; 39 } 40 41 get(const std::string & logger_name)42 std::shared_ptr<logger> get(const std::string& logger_name) 43 { 44 std::lock_guard<Mutex> lock(_mutex); 45 auto found = _loggers.find(logger_name); 46 return found == _loggers.end() ? nullptr : found->second; 47 } 48 49 template<class It> create(const std::string & logger_name,const It & sinks_begin,const It & sinks_end)50 std::shared_ptr<logger> create(const std::string& logger_name, const It& sinks_begin, const It& sinks_end) 51 { 52 std::lock_guard<Mutex> lock(_mutex); 53 throw_if_exists(logger_name); 54 std::shared_ptr<logger> new_logger; 55 if (_async_mode) 56 new_logger = std::make_shared<async_logger>(logger_name, sinks_begin, sinks_end, _async_q_size, _overflow_policy, _worker_warmup_cb, _flush_interval_ms, _worker_teardown_cb); 57 else 58 new_logger = std::make_shared<logger>(logger_name, sinks_begin, sinks_end); 59 60 if (_formatter) 61 new_logger->set_formatter(_formatter); 62 63 if (_err_handler) 64 new_logger->set_error_handler(_err_handler); 65 66 new_logger->set_level(_level); 67 68 69 //Add to registry 70 _loggers[logger_name] = new_logger; 71 return new_logger; 72 } 73 apply_all(std::function<void (std::shared_ptr<logger>)> fun)74 void apply_all(std::function<void(std::shared_ptr<logger>)> fun) 75 { 76 std::lock_guard<Mutex> lock(_mutex); 77 for (auto &l : _loggers) 78 fun(l.second); 79 } 80 drop(const std::string & logger_name)81 void drop(const std::string& logger_name) 82 { 83 std::lock_guard<Mutex> lock(_mutex); 84 _loggers.erase(logger_name); 85 } 86 drop_all()87 void drop_all() 88 { 89 std::lock_guard<Mutex> lock(_mutex); 90 _loggers.clear(); 91 } create(const std::string & logger_name,sinks_init_list sinks)92 std::shared_ptr<logger> create(const std::string& logger_name, sinks_init_list sinks) 93 { 94 return create(logger_name, sinks.begin(), sinks.end()); 95 } 96 create(const std::string & logger_name,sink_ptr sink)97 std::shared_ptr<logger> create(const std::string& logger_name, sink_ptr sink) 98 { 99 return create(logger_name, { sink }); 100 } 101 102 formatter(formatter_ptr f)103 void formatter(formatter_ptr f) 104 { 105 std::lock_guard<Mutex> lock(_mutex); 106 _formatter = f; 107 for (auto& l : _loggers) 108 l.second->set_formatter(_formatter); 109 } 110 set_pattern(const std::string & pattern)111 void set_pattern(const std::string& pattern) 112 { 113 std::lock_guard<Mutex> lock(_mutex); 114 _formatter = std::make_shared<pattern_formatter>(pattern); 115 for (auto& l : _loggers) 116 l.second->set_formatter(_formatter); 117 } 118 set_level(level::level_enum log_level)119 void set_level(level::level_enum log_level) 120 { 121 std::lock_guard<Mutex> lock(_mutex); 122 for (auto& l : _loggers) 123 l.second->set_level(log_level); 124 _level = log_level; 125 } 126 set_error_handler(log_err_handler handler)127 void set_error_handler(log_err_handler handler) 128 { 129 for (auto& l : _loggers) 130 l.second->set_error_handler(handler); 131 _err_handler = handler; 132 } 133 set_async_mode(size_t q_size,const async_overflow_policy overflow_policy,const std::function<void ()> & worker_warmup_cb,const std::chrono::milliseconds & flush_interval_ms,const std::function<void ()> & worker_teardown_cb)134 void set_async_mode(size_t q_size, const async_overflow_policy overflow_policy, const std::function<void()>& worker_warmup_cb, const std::chrono::milliseconds& flush_interval_ms, const std::function<void()>& worker_teardown_cb) 135 { 136 std::lock_guard<Mutex> lock(_mutex); 137 _async_mode = true; 138 _async_q_size = q_size; 139 _overflow_policy = overflow_policy; 140 _worker_warmup_cb = worker_warmup_cb; 141 _flush_interval_ms = flush_interval_ms; 142 _worker_teardown_cb = worker_teardown_cb; 143 } 144 set_sync_mode()145 void set_sync_mode() 146 { 147 std::lock_guard<Mutex> lock(_mutex); 148 _async_mode = false; 149 } 150 instance()151 static registry_t<Mutex>& instance() 152 { 153 static registry_t<Mutex> s_instance; 154 return s_instance; 155 } 156 157 private: 158 registry_t<Mutex>() {} 159 registry_t<Mutex>(const registry_t<Mutex>&) = delete; 160 registry_t<Mutex>& operator=(const registry_t<Mutex>&) = delete; 161 throw_if_exists(const std::string & logger_name)162 void throw_if_exists(const std::string &logger_name) 163 { 164 if (_loggers.find(logger_name) != _loggers.end()) 165 throw spdlog_ex("logger with name '" + logger_name + "' already exists"); 166 } 167 Mutex _mutex; 168 std::unordered_map <std::string, std::shared_ptr<logger>> _loggers; 169 formatter_ptr _formatter; 170 level::level_enum _level = level::info; 171 log_err_handler _err_handler; 172 bool _async_mode = false; 173 size_t _async_q_size = 0; 174 async_overflow_policy _overflow_policy = async_overflow_policy::block_retry; 175 std::function<void()> _worker_warmup_cb = nullptr; 176 std::chrono::milliseconds _flush_interval_ms; 177 std::function<void()> _worker_teardown_cb = nullptr; 178 }; 179 #ifdef SPDLOG_NO_REGISTRY_MUTEX 180 typedef registry_t<spdlog::details::null_mutex> registry; 181 #else 182 typedef registry_t<std::mutex> registry; 183 #endif 184 } 185 } 186