mirror of
https://github.com/emsesp/EMS-ESP32.git
synced 2025-12-07 08:19:52 +03:00
336 lines
7.7 KiB
C++
336 lines
7.7 KiB
C++
/*
|
|
* uuid-log - Microcontroller logging framework
|
|
* Copyright 2019 Simon Arlott
|
|
*
|
|
* This program is free software: you can redistribute it and/or modify
|
|
* it under the terms of the GNU 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 General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#include <uuid/log.h>
|
|
|
|
#include <Arduino.h>
|
|
|
|
#include <cstdarg>
|
|
#include <cstdint>
|
|
#include <list>
|
|
#include <map>
|
|
#include <memory>
|
|
#include <string>
|
|
#include <utility>
|
|
#include <vector>
|
|
|
|
namespace uuid {
|
|
|
|
namespace log {
|
|
|
|
std::map<Handler *, Level> Logger::handlers_;
|
|
Level Logger::level_ = Level::OFF;
|
|
|
|
Message::Message(uint64_t uptime_ms, Level level, Facility facility, const char * name, const std::string && text)
|
|
: uptime_ms(uptime_ms)
|
|
, level(level)
|
|
, facility(facility)
|
|
, name(name)
|
|
, text(std::move(text)) {
|
|
}
|
|
|
|
Logger::Logger(const char * name, Facility facility)
|
|
: name_(name)
|
|
, facility_(facility){
|
|
|
|
};
|
|
|
|
void Logger::register_handler(Handler * handler, Level level) {
|
|
handlers_[handler] = level;
|
|
refresh_log_level();
|
|
};
|
|
|
|
void Logger::unregister_handler(Handler * handler) {
|
|
handlers_.erase(handler);
|
|
refresh_log_level();
|
|
};
|
|
|
|
Level Logger::get_log_level(const Handler * handler) {
|
|
const auto level = handlers_.find(const_cast<Handler *>(handler));
|
|
|
|
if (level != handlers_.end()) {
|
|
return level->second;
|
|
}
|
|
|
|
return Level::OFF;
|
|
}
|
|
|
|
void Logger::emerg(const char * format, ...) const {
|
|
if (enabled(Level::EMERG)) {
|
|
va_list ap;
|
|
|
|
va_start(ap, format);
|
|
vlog(Level::EMERG, format, ap);
|
|
va_end(ap);
|
|
}
|
|
};
|
|
|
|
void Logger::emerg(const __FlashStringHelper * format, ...) const {
|
|
if (enabled(Level::EMERG)) {
|
|
va_list ap;
|
|
|
|
va_start(ap, format);
|
|
vlog(Level::EMERG, format, ap);
|
|
va_end(ap);
|
|
}
|
|
};
|
|
|
|
void Logger::crit(const char * format, ...) const {
|
|
if (enabled(Level::CRIT)) {
|
|
va_list ap;
|
|
|
|
va_start(ap, format);
|
|
vlog(Level::CRIT, format, ap);
|
|
va_end(ap);
|
|
}
|
|
};
|
|
|
|
void Logger::crit(const __FlashStringHelper * format, ...) const {
|
|
if (enabled(Level::CRIT)) {
|
|
va_list ap;
|
|
|
|
va_start(ap, format);
|
|
vlog(Level::CRIT, format, ap);
|
|
va_end(ap);
|
|
}
|
|
};
|
|
|
|
void Logger::alert(const char * format, ...) const {
|
|
if (enabled(Level::ALERT)) {
|
|
va_list ap;
|
|
|
|
va_start(ap, format);
|
|
vlog(Level::ALERT, format, ap);
|
|
va_end(ap);
|
|
}
|
|
};
|
|
|
|
void Logger::alert(const __FlashStringHelper * format, ...) const {
|
|
if (enabled(Level::ALERT)) {
|
|
va_list ap;
|
|
|
|
va_start(ap, format);
|
|
vlog(Level::ALERT, format, ap);
|
|
va_end(ap);
|
|
}
|
|
};
|
|
void Logger::err(const char * format, ...) const {
|
|
if (enabled(Level::ERR)) {
|
|
va_list ap;
|
|
|
|
va_start(ap, format);
|
|
vlog(Level::ERR, format, ap);
|
|
va_end(ap);
|
|
}
|
|
};
|
|
|
|
void Logger::err(const __FlashStringHelper * format, ...) const {
|
|
if (enabled(Level::ERR)) {
|
|
va_list ap;
|
|
|
|
va_start(ap, format);
|
|
vlog(Level::ERR, format, ap);
|
|
va_end(ap);
|
|
}
|
|
};
|
|
|
|
void Logger::warning(const char * format, ...) const {
|
|
if (enabled(Level::WARNING)) {
|
|
va_list ap;
|
|
|
|
va_start(ap, format);
|
|
vlog(Level::WARNING, format, ap);
|
|
va_end(ap);
|
|
}
|
|
};
|
|
|
|
void Logger::warning(const __FlashStringHelper * format, ...) const {
|
|
if (enabled(Level::WARNING)) {
|
|
va_list ap;
|
|
|
|
va_start(ap, format);
|
|
vlog(Level::WARNING, format, ap);
|
|
va_end(ap);
|
|
}
|
|
};
|
|
|
|
void Logger::notice(const char * format, ...) const {
|
|
if (enabled(Level::NOTICE)) {
|
|
va_list ap;
|
|
|
|
va_start(ap, format);
|
|
vlog(Level::NOTICE, format, ap);
|
|
va_end(ap);
|
|
}
|
|
};
|
|
|
|
void Logger::notice(const __FlashStringHelper * format, ...) const {
|
|
if (enabled(Level::NOTICE)) {
|
|
va_list ap;
|
|
|
|
va_start(ap, format);
|
|
vlog(Level::NOTICE, format, ap);
|
|
va_end(ap);
|
|
}
|
|
};
|
|
|
|
void Logger::info(const char * format, ...) const {
|
|
if (enabled(Level::INFO)) {
|
|
va_list ap;
|
|
|
|
va_start(ap, format);
|
|
vlog(Level::INFO, format, ap);
|
|
va_end(ap);
|
|
}
|
|
};
|
|
|
|
void Logger::info(const __FlashStringHelper * format, ...) const {
|
|
if (enabled(Level::INFO)) {
|
|
va_list ap;
|
|
|
|
va_start(ap, format);
|
|
vlog(Level::INFO, format, ap);
|
|
va_end(ap);
|
|
}
|
|
};
|
|
|
|
void Logger::debug(const char * format, ...) const {
|
|
if (enabled(Level::DEBUG)) {
|
|
va_list ap;
|
|
|
|
va_start(ap, format);
|
|
vlog(Level::DEBUG, format, ap);
|
|
va_end(ap);
|
|
}
|
|
};
|
|
|
|
void Logger::debug(const __FlashStringHelper * format, ...) const {
|
|
if (enabled(Level::DEBUG)) {
|
|
va_list ap;
|
|
|
|
va_start(ap, format);
|
|
vlog(Level::DEBUG, format, ap);
|
|
va_end(ap);
|
|
}
|
|
};
|
|
|
|
void Logger::trace(const char * format, ...) const {
|
|
if (enabled(Level::TRACE)) {
|
|
va_list ap;
|
|
|
|
va_start(ap, format);
|
|
vlog(Level::TRACE, format, ap);
|
|
va_end(ap);
|
|
}
|
|
};
|
|
|
|
void Logger::trace(const __FlashStringHelper * format, ...) const {
|
|
if (enabled(Level::TRACE)) {
|
|
va_list ap;
|
|
|
|
va_start(ap, format);
|
|
vlog(Level::TRACE, format, ap);
|
|
va_end(ap);
|
|
}
|
|
};
|
|
|
|
void Logger::log(Level level, Facility facility, const char * format, ...) const {
|
|
if (level < Level::EMERG) {
|
|
level = Level::EMERG;
|
|
} else if (level > Level::TRACE) {
|
|
level = Level::TRACE;
|
|
}
|
|
|
|
if (enabled(level)) {
|
|
va_list ap;
|
|
|
|
va_start(ap, format);
|
|
vlog(level, facility, format, ap);
|
|
va_end(ap);
|
|
}
|
|
};
|
|
|
|
void Logger::log(Level level, Facility facility, const __FlashStringHelper * format, ...) const {
|
|
if (level < Level::EMERG) {
|
|
level = Level::EMERG;
|
|
} else if (level > Level::TRACE) {
|
|
level = Level::TRACE;
|
|
}
|
|
|
|
if (enabled(level)) {
|
|
va_list ap;
|
|
|
|
va_start(ap, format);
|
|
vlog(level, facility, format, ap);
|
|
va_end(ap);
|
|
}
|
|
};
|
|
|
|
void Logger::vlog(Level level, const char * format, va_list ap) const {
|
|
vlog(level, facility_, format, ap);
|
|
}
|
|
|
|
void Logger::vlog(Level level, Facility facility, const char * format, va_list ap) const {
|
|
std::vector<char> text(MAX_LOG_LENGTH + 1);
|
|
|
|
if (vsnprintf(text.data(), text.size(), format, ap) <= 0) {
|
|
return;
|
|
}
|
|
|
|
dispatch(level, facility, text);
|
|
}
|
|
|
|
void Logger::vlog(Level level, const __FlashStringHelper * format, va_list ap) const {
|
|
vlog(level, facility_, format, ap);
|
|
}
|
|
|
|
void Logger::vlog(Level level, Facility facility, const __FlashStringHelper * format, va_list ap) const {
|
|
std::vector<char> text(MAX_LOG_LENGTH + 1);
|
|
|
|
if (vsnprintf_P(text.data(), text.size(), reinterpret_cast<PGM_P>(format), ap) <= 0) {
|
|
return;
|
|
}
|
|
|
|
dispatch(level, facility, text);
|
|
}
|
|
|
|
void Logger::dispatch(Level level, Facility facility, std::vector<char> & text) const {
|
|
std::shared_ptr<Message> message = std::make_shared<Message>(get_uptime_ms(), level, facility, name_, text.data());
|
|
text.resize(0);
|
|
|
|
for (auto & handler : handlers_) {
|
|
if (level <= handler.second) {
|
|
*handler.first << message;
|
|
}
|
|
}
|
|
}
|
|
|
|
void Logger::refresh_log_level() {
|
|
level_ = Level::OFF;
|
|
|
|
for (auto & handler : handlers_) {
|
|
if (level_ < handler.second) {
|
|
level_ = handler.second;
|
|
}
|
|
}
|
|
}
|
|
|
|
} // namespace log
|
|
|
|
} // namespace uuid
|