"docs/source/git@code.ornl.gov:mantidproject/mantid.git" did not exist on "313ac056c6dd6fb93b44c254154f5af34a50c61e"
Newer
Older
// Mantid Repository : https://github.com/mantidproject/mantid
//
// Copyright © 2018 ISIS Rutherford Appleton Laboratory UKRI,
// NScD Oak Ridge National Laboratory, European Spallation Source
// & Institut Laue - Langevin
// SPDX - License - Identifier: GPL - 3.0 +
#include "MantidKernel/ThreadSafeLogStream.h"
#include <algorithm>
#include <exception>
Nick Draper
committed
#include <iostream>
namespace Mantid {
namespace Kernel {
namespace {
// We only need a single NullStream object
Poco::NullOutputStream NULL_STREAM;
static const std::string PriorityNames_data[] = {
"NOT_USED", "PRIO_FATAL", "PRIO_CRITICAL",
"PRIO_ERROR", "PRIO_WARNING", "PRIO_NOTICE",
"PRIO_INFORMATION", "PRIO_DEBUG", "PRIO_TRACE"};
const std::string *Logger::PriorityNames = PriorityNames_data;
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
/** Constructor
* @param name :: The class name invoking this logger
*/
Logger::Logger(const std::string &name)
: m_log(&Poco::Logger::get(name)),
m_logStream(new ThreadSafeLogStream(*m_log)), m_levelOffset(0),
m_enabled(true) {}
/// Destructor
Logger::~Logger() { delete m_logStream; }
/**
* @param name The new name
*/
void Logger::setName(const std::string &name) {
auto *logger = &Poco::Logger::get(name);
auto *logStream =
new ThreadSafeLogStream(*logger); // don't swap if this throws
using std::swap;
swap(m_log, logger);
swap(m_logStream, logStream);
delete logStream;
}
/** Returns true if the log is enabled
*
* @retval true - logging is enabled
* @retval false - all messages are ignored.
*/
bool Logger::getEnabled() const { return m_enabled; }
/** set if the logging is enabled
*
* @param enabled :: true - logging is enabled, false - all messages are
*ignored.
*/
void Logger::setEnabled(const bool enabled) { m_enabled = enabled; }
/** If the Logger's log level is at least Poco::Message::PRIO_FATAL, creates a
*Message with
* priority Poco::Message::PRIO_FATAL and the given message text and sends it
*to the attached channel.
*
* @param msg :: The message to log.
*/
void Logger::fatal(const std::string &msg) {
log(msg, Poco::Message::PRIO_FATAL);
}
/** If the Logger's log level is at least Poco::Message::PRIO_ERROR, creates a
*Message with priority
* Poco::Message::PRIO_ERROR and the given message text and sends it to the
*attached channel.
*
* @param msg :: The message to log.
*/
void Logger::error(const std::string &msg) {
log(msg, Poco::Message::PRIO_ERROR);
}
/** If the Logger's log level is at least Poco::Message::PRIO_WARNING, creates a
*Message with
* priority Poco::Message::PRIO_WARNING and the given message text and sends it
*to the attached channel.
*
* @param msg :: The message to log.
*/
void Logger::warning(const std::string &msg) {
log(msg, Poco::Message::PRIO_WARNING);
}
/** If the Logger's log level is at least Poco::Message::PRIO_NOTICE, creates a
*Message with
* priority Poco::Message::PRIO_NOTICE and the given message text and sends it
*to the attached channel.
*
* @param msg :: The message to log.
*/
void Logger::notice(const std::string &msg) {
log(msg, Poco::Message::PRIO_NOTICE);
}
/** If the Logger's log level is at least Poco::Message::PRIO_INFORMATION,
*creates a Message with
* priority Poco::Message::PRIO_INFORMATION and the given message text and
*sends it to the
* attached channel.
*
* @param msg :: The message to log.
*/
void Logger::information(const std::string &msg) {
log(msg, Poco::Message::PRIO_INFORMATION);
}
/** If the Logger's log level is at least Poco::Message::PRIO_DEBUG, creates a
*Message with priority
* Poco::Message::PRIO_DEBUG and the given message text and sends it to the
*attached channel.
*
* @param msg :: The message to log.
*/
void Logger::debug(const std::string &msg) {
log(msg, Poco::Message::PRIO_DEBUG);
}
/** Logs the given message at debug level, followed by the data in buffer.
*
* The data in buffer is written in canonical hex+ASCII form:
* Offset (4 bytes) in hexadecimal, followed by sixteen space-separated,
* two column, hexadecimal bytes, followed by the same sixteen bytes as
* ASCII characters.
* For bytes outside the range 32 .. 127, a dot is printed.
* Note all Dump messages go out at Debug message level
*
* @param msg :: The message to log
* @param buffer :: the binary data to log
* @param length :: The length of the binaary data to log
*/
void Logger::dump(const std::string &msg, const void *buffer,
std::size_t length) {
if (m_enabled) {
try {
m_log->dump(msg, buffer, length);
} catch (std::exception &e) {
// failures in logging are not allowed to throw exceptions out of the
// logging class
std::cerr << e.what();
Gigg, Martyn Anthony
committed
}
}
}
/** Returns true if at least the given log level is set.
* @param level :: The logging level it is best to use the Logger::Priority
* enum (7=debug, 6=information, 4=warning, 3=error, 2=critical, 1=fatal)
* @return true if at least the given log level is set.
*/
bool Logger::is(int level) const {
bool retVal = false;
try {
retVal = m_log->is(level);
} catch (std::exception &e) {
// failures in logging are not allowed to throw exceptions out of the
// logging class
std::cerr << e.what();
}
return retVal;
}
void Logger::setLevel(int level) {
try {
m_log->setLevel(level);
} catch (std::exception &e) {
// failures in logging are not allowed to throw exceptions out of the
// logging class
std::cerr << e.what();
}
}
/// Sets the Logger's log level using a symbolic value.
///
/// @param level :: Valid values are: fatal, critical, error, warning, notice,
/// information, debug
void Logger::setLevel(const std::string &level) {
try {
m_log->setLevel(level);
} catch (std::exception &e) {
// failures in logging are not allowed to throw exceptions out of the
// logging class
std::cerr << e.what();
}
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
}
int Logger::getLevel() const { return m_log->getLevel(); }
/** This class implements an ostream interface to the Logger for fatal messages.
*
* The stream's buffer appends all characters written to it
* to a string. As soon as a CR or LF (std::endl) is written,
* the string is sent to the Logger.
* @returns an std::ostream reference.
*/
std::ostream &Logger::fatal() { return getLogStream(Priority::PRIO_FATAL); }
/** This class implements an ostream interface to the Logger for error messages.
*
* The stream's buffer appends all characters written to it
* to a string. As soon as a CR or LF (std::endl) is written,
* the string is sent to the Logger.
* @returns an std::ostream reference.
*/
std::ostream &Logger::error() { return getLogStream(Priority::PRIO_ERROR); }
/** This class implements an ostream interface to the Logger for warning
*messages.
*
* The stream's buffer appends all characters written to it
* to a string. As soon as a CR or LF (std::endl) is written,
* the string is sent to the Logger.
* @returns an std::ostream reference.
*/
std::ostream &Logger::warning() { return getLogStream(Priority::PRIO_WARNING); }
/** This class implements an ostream interface to the Logger for notice
*messages.
*
* The stream's buffer appends all characters written to it
* to a string. As soon as a CR or LF (std::endl) is written,
* the string is sent to the Logger.
* @returns an std::ostream reference.
*/
std::ostream &Logger::notice() { return getLogStream(Priority::PRIO_NOTICE); }
/** This class implements an ostream interface to the Logger for information
*messages.
*
* The stream's buffer appends all characters written to it
* to a string. As soon as a CR or LF (std::endl) is written,
* the string is sent to the Logger.
* @returns an std::ostream reference.
*/
std::ostream &Logger::information() {
return getLogStream(Priority::PRIO_INFORMATION);
}
/** This class implements an ostream interface to the Logger for debug messages.
*
* The stream's buffer appends all characters written to it
* to a string. As soon as a CR or LF (std::endl) is written,
* the string is sent to the Logger.
* @returns an std::ostream reference.
*/
std::ostream &Logger::debug() { return getLogStream(Priority::PRIO_DEBUG); }
/** Shuts down the logging framework and releases all Loggers.
* Static method.
*/
void Logger::shutdown() {
try {
// Release the POCO loggers
Poco::Logger::shutdown();
} catch (std::exception &e) {
// failures in logging are not allowed to throw exceptions out of the
// logging class
std::cerr << e.what();
}
/** Sets the log level for all Loggers created so far, including the root
* logger.
* @param level :: the priority level to set for the loggers
*/
void Logger::setLevelForAll(const int level) {
// "" is the root logger
Poco::Logger::setLevel("", level);
}
/**
* @param message :: The message to log
* @param priority :: The priority level
*/
void Logger::log(const std::string &message, Logger::Priority priority) {
if (!m_enabled)
return;
try {
switch (applyLevelOffset(priority)) {
case Poco::Message::PRIO_FATAL:
m_log->fatal(message);
break;
case Poco::Message::PRIO_CRITICAL:
m_log->critical(message);
case Poco::Message::PRIO_ERROR:
m_log->error(message);
break;
case Poco::Message::PRIO_WARNING:
m_log->warning(message);
break;
case Poco::Message::PRIO_NOTICE:
m_log->notice(message);
break;
case Poco::Message::PRIO_INFORMATION:
m_log->information(message);
break;
case Poco::Message::PRIO_DEBUG:
m_log->debug(message);
break;
case Poco::Message::PRIO_TRACE:
m_log->trace(message);
break;
default:
}
} catch (std::exception &e) {
// Failures in logging are not allowed to throw exceptions out of the
// logging class
std::cerr << "Error in logging framework: " << e.what();
}
* Log a given message at a given priority
* @param priority :: The priority level
* @return :: the stream
*/
std::ostream &Logger::getLogStream(Logger::Priority priority) {
if (!m_enabled)
return NULL_STREAM;
switch (applyLevelOffset(priority)) {
case Poco::Message::PRIO_FATAL:
return m_logStream->fatal();
case Poco::Message::PRIO_CRITICAL:
return m_logStream->critical();
case Poco::Message::PRIO_ERROR:
return m_logStream->error();
case Poco::Message::PRIO_WARNING:
return m_logStream->warning();
case Poco::Message::PRIO_NOTICE:
return m_logStream->notice();
case Poco::Message::PRIO_INFORMATION:
return m_logStream->information();
case Poco::Message::PRIO_DEBUG:
return m_logStream->debug();
}
/**
* Adjust a log priority level based off the m_levelOffset
* @param proposedLevel :: The proposed level
* @returns The offseted level
*/
Logger::Priority Logger::applyLevelOffset(Logger::Priority proposedLevel) {
int retVal = proposedLevel;
// fast exit is offset is 0
if (m_levelOffset == 0) {
return proposedLevel;
} else {
retVal += m_levelOffset;
if (retVal < static_cast<int>(Priority::PRIO_FATAL)) {
retVal = Priority::PRIO_FATAL;
} else if (retVal > static_cast<int>(Priority::PRIO_TRACE)) {
retVal = Priority::PRIO_TRACE;
// Logger::Priority p(retVal);
return static_cast<Logger::Priority>(retVal);
}
/**
* Sets the Logger's log offset level.
* @param level :: The level offset to use
*/
void Logger::setLevelOffset(int level) { m_levelOffset = level; }
/**
* Gets the Logger's log offset level.
* @returns The offset level
*/ /// Gets the Logger's log offset level.
int Logger::getLevelOffset() const { return m_levelOffset; }
Gigg, Martyn Anthony
committed
} // namespace Kernel
Russell Taylor
committed
} // Namespace Mantid