196 lines
4.8 KiB
C++
196 lines
4.8 KiB
C++
/*
|
||
PIP - Platform Independent Primitives
|
||
High-level log
|
||
Ivan Pelipenko peri4ko@yandex.ru
|
||
|
||
This program is free software: you can redistribute it and/or modify
|
||
it under the terms of the GNU Lesser General Public License as published by
|
||
the Free Software Foundation, either version 3 of the License, or
|
||
(at your option) any later version.
|
||
|
||
This program is distributed in the hope that it will be useful,
|
||
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||
GNU Lesser General Public License for more details.
|
||
|
||
You should have received a copy of the GNU Lesser General Public License
|
||
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||
*/
|
||
|
||
#include "pilog.h"
|
||
|
||
#include "pidir.h"
|
||
#include "piliterals_string.h"
|
||
#include "piliterals_time.h"
|
||
#include "pitime.h"
|
||
|
||
|
||
//! \class PILog pilog.h
|
||
//! \details
|
||
//! \~english \section PILog_sec0 Synopsis
|
||
//! \~russian \section PILog_sec0 Краткий обзор
|
||
//! \~english
|
||
//! This class provides log with optional file and console output.
|
||
//!
|
||
//! \~russian
|
||
//! Этот класс предоставляет лог с опциональным выводом в файл и консоль.
|
||
//!
|
||
|
||
|
||
PILog::PILog(): PIThread(), log_ts(&log_file) {
|
||
setName("PILog");
|
||
split_time = 8_h;
|
||
timestamp_format = "yyyy-MM-dd hh:mm:ss.zzz";
|
||
setLineFormat("t - c: m");
|
||
id_by_cat[Level::Info] = PICout::registerExternalBufferID();
|
||
id_by_cat[Level::Debug] = PICout::registerExternalBufferID();
|
||
id_by_cat[Level::Warning] = PICout::registerExternalBufferID();
|
||
id_by_cat[Level::Error] = PICout::registerExternalBufferID();
|
||
CONNECTU(PICout::Notifier::object(), finished, this, coutDone);
|
||
}
|
||
|
||
|
||
PILog::~PILog() {
|
||
stop();
|
||
}
|
||
|
||
|
||
void PILog::setDir(const PIString & d) {
|
||
stopAndWait();
|
||
log_dir = d;
|
||
if (output[File]) {
|
||
PIDir::make(log_dir);
|
||
newFile();
|
||
}
|
||
start();
|
||
}
|
||
|
||
|
||
void PILog::setLineFormat(const PIString & f) {
|
||
line_format = f;
|
||
line_format_p = line_format;
|
||
line_format_p.replace("t", "${t}").replace("c", "${c}").replace("m", "${m}");
|
||
}
|
||
|
||
|
||
void PILog::setLevel(Level l) {
|
||
max_level = l;
|
||
}
|
||
|
||
|
||
PICout PILog::error(PIObject * context) {
|
||
return makePICout(context, Level::Error);
|
||
}
|
||
|
||
|
||
PICout PILog::warning(PIObject * context) {
|
||
return makePICout(context, Level::Warning);
|
||
}
|
||
|
||
|
||
PICout PILog::info(PIObject * context) {
|
||
return makePICout(context, Level::Info);
|
||
}
|
||
|
||
|
||
PICout PILog::debug(PIObject * context) {
|
||
return makePICout(context, Level::Debug);
|
||
}
|
||
|
||
|
||
void PILog::stop() {
|
||
while (true) {
|
||
log_mutex.lock();
|
||
bool done = queue.isEmpty();
|
||
log_mutex.unlock();
|
||
if (done) break;
|
||
piMinSleep();
|
||
}
|
||
PIThread::stopAndWait();
|
||
}
|
||
|
||
|
||
void PILog::coutDone(int id, PIString * buffer) {
|
||
if (!buffer) return;
|
||
if (!id_by_cat.containsValue(id)) return;
|
||
auto cat = id_by_cat.key(id, PILog::Level::Debug);
|
||
if (cat > max_level) return;
|
||
enqueue(*buffer, cat);
|
||
delete buffer;
|
||
}
|
||
|
||
|
||
PICout PILog::makePICout(PIObject * context, Level cat) {
|
||
auto buffer = new PIString();
|
||
if (context) {
|
||
*buffer = "["_a + context->className();
|
||
if (context->name().isNotEmpty()) *buffer += " \"" + context->name() + "\"";
|
||
*buffer += "] ";
|
||
}
|
||
return PICout::withExternalBuffer(buffer, id_by_cat.value(cat), PICoutManipulators::AddSpaces);
|
||
}
|
||
|
||
|
||
void PILog::enqueue(const PIString & msg, Level cat) {
|
||
auto t = PIDateTime::fromSystemTime(PISystemTime::current());
|
||
PIMutexLocker ml(log_mutex);
|
||
queue.enqueue({cat, t, msg});
|
||
}
|
||
|
||
|
||
PIString PILog::entryToString(const Entry & e) const {
|
||
static PIStringList categories{"error", "warn ", "info ", "debug"};
|
||
PIString t = e.time.toString(timestamp_format);
|
||
PIString ret = line_format_p;
|
||
ret.replace("${t}", t).replace("${c}", categories[static_cast<int>(e.cat)]).replace("${m}", e.msg);
|
||
return ret;
|
||
}
|
||
|
||
|
||
void PILog::newFile() {
|
||
PIString aname = log_name;
|
||
if (aname.isNotEmpty()) aname += "__";
|
||
log_file.open(log_dir + "/" + aname + PIDateTime::current().toString("yyyy_MM_dd__hh_mm_ss") + ".log." +
|
||
PIString::fromNumber(++part_number),
|
||
PIIODevice::ReadWrite);
|
||
}
|
||
|
||
|
||
void PILog::run() {
|
||
if (output[File]) {
|
||
if (split_tm.elapsed() >= split_time) {
|
||
split_tm.reset();
|
||
newFile();
|
||
}
|
||
}
|
||
log_mutex.lock();
|
||
if (queue.isEmpty()) {
|
||
log_mutex.unlock();
|
||
piMSleep(20);
|
||
return;
|
||
}
|
||
log_mutex.unlock();
|
||
while (true) {
|
||
log_mutex.lock();
|
||
if (queue.isEmpty()) {
|
||
log_mutex.unlock();
|
||
return;
|
||
}
|
||
auto qi = queue.dequeue();
|
||
log_mutex.unlock();
|
||
auto str = entryToString(qi);
|
||
if (log_file.isOpened()) log_ts << str << "\n";
|
||
if (output[Console]) {
|
||
PICout out(qi.cat == Level::Error ? piCerr : piCout);
|
||
if (color_console) {
|
||
switch (qi.cat) {
|
||
case Level::Error: out << PICoutManipulators::Red; break;
|
||
case Level::Warning: out << PICoutManipulators::Yellow; break;
|
||
default: break;
|
||
}
|
||
}
|
||
out << str;
|
||
}
|
||
}
|
||
}
|