2018-10-24 22:15:55 +03:00
|
|
|
//
|
2019-01-01 01:02:34 +03:00
|
|
|
// Copyright Aliaksei Levin (levlam@telegram.org), Arseny Smirnov (arseny30@gmail.com) 2014-2019
|
2018-10-24 22:15:55 +03:00
|
|
|
//
|
|
|
|
// Distributed under the Boost Software License, Version 1.0. (See accompanying
|
|
|
|
// file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
|
|
|
|
//
|
|
|
|
#include "td/telegram/Logging.h"
|
|
|
|
|
2018-12-26 23:58:18 +03:00
|
|
|
#include "td/telegram/ConfigManager.h"
|
2019-01-14 21:46:04 +03:00
|
|
|
#include "td/telegram/FileReferenceManager.h"
|
2018-11-16 00:25:08 +03:00
|
|
|
#include "td/telegram/files/FileGcWorker.h"
|
2018-10-24 22:15:55 +03:00
|
|
|
#include "td/telegram/files/FileManager.h"
|
2018-10-28 20:30:47 +03:00
|
|
|
#include "td/telegram/net/ConnectionCreator.h"
|
2018-11-11 15:58:52 +03:00
|
|
|
#include "td/telegram/NotificationManager.h"
|
2018-10-24 22:15:55 +03:00
|
|
|
#include "td/telegram/Td.h"
|
2018-11-15 20:25:54 +03:00
|
|
|
#include "td/telegram/UpdatesManager.h"
|
2018-10-24 22:15:55 +03:00
|
|
|
|
2019-02-04 17:02:21 +03:00
|
|
|
#include "td/db/binlog/BinlogEvent.h"
|
2018-10-24 22:15:55 +03:00
|
|
|
|
2019-02-04 17:02:21 +03:00
|
|
|
#include "td/net/GetHostByNameActor.h"
|
|
|
|
#include "td/net/TransparentProxy.h"
|
2018-10-24 22:15:55 +03:00
|
|
|
|
|
|
|
#include "td/utils/FileLog.h"
|
|
|
|
#include "td/utils/logging.h"
|
|
|
|
#include "td/utils/misc.h"
|
|
|
|
|
2019-03-27 21:48:21 +03:00
|
|
|
#include <atomic>
|
2018-12-11 23:18:58 +03:00
|
|
|
#include <map>
|
2018-10-24 22:15:55 +03:00
|
|
|
#include <mutex>
|
|
|
|
|
|
|
|
namespace td {
|
|
|
|
|
|
|
|
static std::mutex logging_mutex;
|
|
|
|
static FileLog file_log;
|
|
|
|
static TsLog ts_log(&file_log);
|
|
|
|
static NullLog null_log;
|
|
|
|
|
|
|
|
#define ADD_TAG(tag) \
|
|
|
|
{ #tag, &VERBOSITY_NAME(tag) }
|
2018-11-15 20:25:54 +03:00
|
|
|
static const std::map<Slice, int *> log_tags{
|
2019-03-26 16:24:29 +03:00
|
|
|
ADD_TAG(td_init), ADD_TAG(update_file), ADD_TAG(connections), ADD_TAG(binlog),
|
|
|
|
ADD_TAG(proxy), ADD_TAG(net_query), ADD_TAG(td_requests), ADD_TAG(dc),
|
|
|
|
ADD_TAG(files), ADD_TAG(mtproto), ADD_TAG(raw_mtproto), ADD_TAG(fd),
|
|
|
|
ADD_TAG(actor), ADD_TAG(sqlite), ADD_TAG(notifications), ADD_TAG(get_difference),
|
|
|
|
ADD_TAG(file_gc), ADD_TAG(config_recoverer), ADD_TAG(dns_resolver), ADD_TAG(file_references)};
|
2018-10-24 22:15:55 +03:00
|
|
|
#undef ADD_TAG
|
|
|
|
|
|
|
|
Status Logging::set_current_stream(td_api::object_ptr<td_api::LogStream> stream) {
|
|
|
|
if (stream == nullptr) {
|
|
|
|
return Status::Error("Log stream must not be empty");
|
|
|
|
}
|
|
|
|
|
|
|
|
std::lock_guard<std::mutex> lock(logging_mutex);
|
|
|
|
switch (stream->get_id()) {
|
|
|
|
case td_api::logStreamDefault::ID:
|
|
|
|
log_interface = default_log_interface;
|
|
|
|
return Status::OK();
|
|
|
|
case td_api::logStreamFile::ID: {
|
|
|
|
auto file_stream = td_api::move_object_as<td_api::logStreamFile>(stream);
|
|
|
|
auto max_log_file_size = file_stream->max_file_size_;
|
|
|
|
if (max_log_file_size <= 0) {
|
|
|
|
return Status::Error("Max log file size should be positive");
|
|
|
|
}
|
|
|
|
|
|
|
|
TRY_STATUS(file_log.init(file_stream->path_, max_log_file_size));
|
2019-03-27 21:48:21 +03:00
|
|
|
std::atomic_thread_fence(std::memory_order_release); // better than nothing
|
2018-10-24 22:15:55 +03:00
|
|
|
log_interface = &ts_log;
|
|
|
|
return Status::OK();
|
|
|
|
}
|
|
|
|
case td_api::logStreamEmpty::ID:
|
|
|
|
log_interface = &null_log;
|
|
|
|
return Status::OK();
|
|
|
|
default:
|
|
|
|
UNREACHABLE();
|
|
|
|
return Status::OK();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Result<td_api::object_ptr<td_api::LogStream>> Logging::get_current_stream() {
|
|
|
|
std::lock_guard<std::mutex> lock(logging_mutex);
|
|
|
|
if (log_interface == default_log_interface) {
|
|
|
|
return td_api::make_object<td_api::logStreamDefault>();
|
|
|
|
}
|
|
|
|
if (log_interface == &null_log) {
|
|
|
|
return td_api::make_object<td_api::logStreamEmpty>();
|
|
|
|
}
|
|
|
|
if (log_interface == &ts_log) {
|
|
|
|
return td_api::make_object<td_api::logStreamFile>(file_log.get_path().str(), file_log.get_rotate_threshold());
|
|
|
|
}
|
|
|
|
return Status::Error("Log stream is unrecognized");
|
|
|
|
}
|
|
|
|
|
|
|
|
Status Logging::set_verbosity_level(int new_verbosity_level) {
|
|
|
|
std::lock_guard<std::mutex> lock(logging_mutex);
|
|
|
|
if (0 <= new_verbosity_level && new_verbosity_level <= VERBOSITY_NAME(NEVER)) {
|
|
|
|
SET_VERBOSITY_LEVEL(VERBOSITY_NAME(FATAL) + new_verbosity_level);
|
|
|
|
return Status::OK();
|
|
|
|
}
|
|
|
|
|
|
|
|
return Status::Error("Wrong new verbosity level specified");
|
|
|
|
}
|
|
|
|
|
|
|
|
int Logging::get_verbosity_level() {
|
|
|
|
std::lock_guard<std::mutex> lock(logging_mutex);
|
|
|
|
return GET_VERBOSITY_LEVEL();
|
|
|
|
}
|
|
|
|
|
|
|
|
vector<string> Logging::get_tags() {
|
|
|
|
return transform(log_tags, [](auto &tag) { return tag.first.str(); });
|
|
|
|
}
|
|
|
|
|
|
|
|
Status Logging::set_tag_verbosity_level(Slice tag, int new_verbosity_level) {
|
|
|
|
auto it = log_tags.find(tag);
|
|
|
|
if (it == log_tags.end()) {
|
|
|
|
return Status::Error("Log tag is not found");
|
|
|
|
}
|
|
|
|
|
|
|
|
std::lock_guard<std::mutex> lock(logging_mutex);
|
|
|
|
*it->second = clamp(new_verbosity_level, 1, VERBOSITY_NAME(NEVER));
|
|
|
|
return Status::OK();
|
|
|
|
}
|
|
|
|
|
|
|
|
Result<int> Logging::get_tag_verbosity_level(Slice tag) {
|
|
|
|
auto it = log_tags.find(tag);
|
|
|
|
if (it == log_tags.end()) {
|
|
|
|
return Status::Error("Log tag is not found");
|
|
|
|
}
|
|
|
|
|
|
|
|
std::lock_guard<std::mutex> lock(logging_mutex);
|
|
|
|
return *it->second;
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace td
|