PcapPlusPlus  Next
Logger.h
Go to the documentation of this file.
1 #pragma once
2 
3 #include <cstdio>
4 #include <cstdint>
5 #include <memory>
6 #include <array>
7 #include <mutex>
8 #include <ostream>
9 #include <sstream>
10 #include <functional>
11 #include "DeprecationUtils.h"
12 #include "ObjectPool.h"
13 
14 #ifndef LOG_MODULE
15 # define LOG_MODULE UndefinedLogModule
16 #endif
17 
18 // Use __FILE_NAME__ to avoid leaking complete full path
19 #ifdef __FILE_NAME__
20 # define PCAPPP_FILENAME __FILE_NAME__
21 #else
22 # define PCAPPP_FILENAME __FILE__
23 #endif
24 
26 
27 // Compile time log levels.
28 // Allows for conditional removal of unwanted log calls at compile time.
29 #define PCPP_LOG_LEVEL_OFF 0
30 #define PCPP_LOG_LEVEL_ERROR 1
31 #define PCPP_LOG_LEVEL_WARN 2
32 #define PCPP_LOG_LEVEL_INFO 3
33 #define PCPP_LOG_LEVEL_DEBUG 4
34 
35 // All log messages built via a PCPP_LOG_* macro below the PCPP_ACTIVE_LOG_LEVEL will be removed at compile time.
36 // Uses the PCPP_ACTIVE_LOG_LEVEL if it is defined, otherwise defaults to PCAP_LOG_LEVEL_DEBUG
37 #ifndef PCPP_ACTIVE_LOG_LEVEL
38 # define PCPP_ACTIVE_LOG_LEVEL PCPP_LOG_LEVEL_DEBUG
39 #endif // !PCPP_ACTIVE_LOG_LEVEL
40 
43 namespace pcpp
44 {
48  std::string getErrorString(int errnum);
49 
51  enum LogModule : uint8_t
52  {
53  UndefinedLogModule,
115  NumOfLogModules
116  };
117 
121  struct LogSource
122  {
124  constexpr LogSource() = default;
125 
129  {}
130 
136  constexpr LogSource(LogModule logModule, const char* file, const char* function, int line)
138  {}
139 
140  const char* file = nullptr;
141  const char* function = nullptr;
142  int line = 0;
143  LogModule logModule = UndefinedLogModule;
144  };
145 
148  enum class LogLevel : uint8_t
149  {
150  Off = PCPP_LOG_LEVEL_OFF,
151  Error = PCPP_LOG_LEVEL_ERROR,
152  Warn = PCPP_LOG_LEVEL_WARN,
153  Info = PCPP_LOG_LEVEL_INFO,
154  Debug = PCPP_LOG_LEVEL_DEBUG
155  };
156 
157  inline std::ostream& operator<<(std::ostream& ostr, LogLevel lvl)
158  {
159  return ostr << static_cast<std::underlying_type_t<LogLevel>>(lvl);
160  }
161 
162  // Forward declaration
163  class Logger;
164 
165  namespace internal
166  {
170  {
171  public:
172  friend class pcpp::Logger;
173 
175  LogContext() = default;
176 
180  explicit LogContext(LogLevel level, LogSource const& source = {}) : m_Source(source), m_Level(level)
181  {}
182 
186  void init(LogLevel level, LogSource const& source)
187  {
188  m_Source = source;
189  m_Level = level;
190  m_Stream.clear();
191  m_Stream.str({});
192  }
193 
197  template <class T> LogContext& operator<<(T const& value)
198  {
199  m_Stream << value;
200  return *this;
201  }
202 
203  private:
204  std::ostringstream m_Stream;
205  LogSource m_Source;
206  LogLevel m_Level = LogLevel::Info;
207  };
208  } // namespace internal
209 
227  class Logger
228  {
229  public:
230  Logger(const Logger&) = delete;
231  Logger& operator=(const Logger&) = delete;
232 
233  // Deprecated, Use the LogLevel in the pcpp namespace instead.
234  using LogLevel = pcpp::LogLevel;
235  PCPP_DEPRECATED("Use the LogLevel in the pcpp namespace instead.")
236  static const LogLevel Error = LogLevel::Error;
237  PCPP_DEPRECATED("Use the LogLevel in the pcpp namespace instead.")
238  static const LogLevel Info = LogLevel::Info;
239  PCPP_DEPRECATED("Use the LogLevel in the pcpp namespace instead.")
240  static const LogLevel Debug = LogLevel::Debug;
241 
250  using LogPrinter = std::function<void(LogLevel logLevel, const std::string& logMessage, const std::string& file,
251  const std::string& method, const int line)>;
252 
256  static std::string logLevelAsString(LogLevel logLevel);
257 
262  {
263  return m_LogModulesArray[module];
264  }
265 
269  void setLogLevel(LogModule module, LogLevel level)
270  {
271  m_LogModulesArray[module] = level;
272  }
273 
277  bool isDebugEnabled(LogModule module) const
278  {
279  return m_LogModulesArray[module] == LogLevel::Debug;
280  }
281 
286  bool shouldLog(LogLevel level, LogModule module) const
287  {
288  return level != LogLevel::Off && m_LogModulesArray[module] >= level;
289  }
290 
294  {
295  for (int i = 1; i < NumOfLogModules; i++)
296  {
297  m_LogModulesArray[i] = level;
298  }
299  }
300 
303  void setLogPrinter(LogPrinter printer)
304  {
305  m_LogPrinter = printer;
306  }
307 
310 
312  std::string getLastError() const
313  {
314  std::lock_guard<std::mutex> const lock(m_LastErrorMtx);
315  return m_LastError;
316  }
317 
320  {
321  m_LogsEnabled = false;
322  }
323 
325  void enableLogs()
326  {
327  m_LogsEnabled = true;
328  }
329 
332  bool logsEnabled() const
333  {
334  return m_LogsEnabled;
335  }
336 
344  void useContextPooling(bool enabled, std::size_t preallocate = 2, std::size_t maxPoolSize = 10)
345  {
346  m_UseContextPooling = enabled;
347 
348  if (m_UseContextPooling)
349  {
350  m_LogContextPool.setMaxSize(maxPoolSize);
351 
352  if (preallocate > 0)
353  {
354  m_LogContextPool.preallocate(preallocate);
355  }
356  }
357  else
358  {
359  // Clear the pool if we're disabling pooling.
360  m_LogContextPool.clear();
361  }
362  }
363 
367  static Logger& getInstance()
368  {
369  static Logger instance;
370  return instance;
371  }
372 
375  std::unique_ptr<internal::LogContext> createLogContext();
376 
381  std::unique_ptr<internal::LogContext> createLogContext(LogLevel level, LogSource const& source = {});
382 
387  void emit(LogSource const& source, LogLevel level, std::string const& message);
388 
391  void emit(std::unique_ptr<internal::LogContext> message);
392 
393  private:
394  bool m_LogsEnabled;
395  std::array<LogLevel, NumOfLogModules> m_LogModulesArray{};
396  LogPrinter m_LogPrinter;
397 
398  mutable std::mutex m_LastErrorMtx;
399  std::string m_LastError;
400 
401  bool m_UseContextPooling = true;
402  // Keep a maximum of 10 LogContext objects in the pool.
403  internal::DynamicObjectPool<internal::LogContext> m_LogContextPool{ 10, 2 };
404 
405  // private c'tor - this class is a singleton
406  Logger();
407  };
408 
409 } // namespace pcpp
410 
411 #define PCPP_LOG(level, message) \
412  do \
413  { \
414  auto& logger = pcpp::Logger::getInstance(); \
415  if (logger.shouldLog(level, LOG_MODULE)) \
416  { \
417  auto ctx = \
418  logger.createLogContext(level, pcpp::LogSource(LOG_MODULE, PCAPPP_FILENAME, __FUNCTION__, __LINE__)); \
419  (*ctx) << message; \
420  logger.emit(std::move(ctx)); \
421  } \
422  } while (0)
423 
424 #if PCPP_ACTIVE_LOG_LEVEL >= PCPP_LOG_LEVEL_DEBUG
425 # define PCPP_LOG_DEBUG(message) PCPP_LOG(pcpp::LogLevel::Debug, message)
426 #else
427 # define PCPP_LOG_DEBUG(message) (void)0
428 #endif
429 
430 #if PCPP_ACTIVE_LOG_LEVEL >= PCPP_LOG_LEVEL_WARN
431 # define PCPP_LOG_WARN(message) PCPP_LOG(pcpp::LogLevel::Warn, message)
432 #else
433 # define PCPP_LOG_WARN(message) (void)0
434 #endif
435 
436 #if PCPP_ACTIVE_LOG_LEVEL >= PCPP_LOG_LEVEL_INFO
437 # define PCPP_LOG_INFO(message) PCPP_LOG(pcpp::LogLevel::Info, message)
438 #else
439 # define PCPP_LOG_INFO(message) (void)0
440 #endif
441 
442 #if PCPP_ACTIVE_LOG_LEVEL >= PCPP_LOG_LEVEL_ERROR
443 # define PCPP_LOG_ERROR(message) PCPP_LOG(pcpp::LogLevel::Error, message)
444 #else
445 # define PCPP_LOG_ERROR(message) (void)0
446 #endif
Definition: Logger.h:228
void setLogPrinter(LogPrinter printer)
Definition: Logger.h:303
void enableLogs()
Enable logs in all PcapPlusPlus modules.
Definition: Logger.h:325
void emit(LogSource const &source, LogLevel level, std::string const &message)
Directly emits a log message bypassing all level checks.
void setLogLevel(LogModule module, LogLevel level)
Definition: Logger.h:269
std::string getLastError() const
Definition: Logger.h:312
LogLevel getLogLevel(LogModule module)
Definition: Logger.h:261
void useContextPooling(bool enabled, std::size_t preallocate=2, std::size_t maxPoolSize=10)
Controls if the logger should use a pool of LogContext objects.
Definition: Logger.h:344
bool shouldLog(LogLevel level, LogModule module) const
Check whether a log level should be emitted by the logger.
Definition: Logger.h:286
bool logsEnabled() const
Definition: Logger.h:332
void resetLogPrinter()
Set the log printer back to the default printer.
static Logger & getInstance()
Definition: Logger.h:367
std::unique_ptr< internal::LogContext > createLogContext()
Creates a new LogContext with Info level and no source.
void emit(std::unique_ptr< internal::LogContext > message)
Directly emits a log message bypassing all level checks.
void suppressLogs()
Suppress logs in all PcapPlusPlus modules.
Definition: Logger.h:319
static std::string logLevelAsString(LogLevel logLevel)
std::unique_ptr< internal::LogContext > createLogContext(LogLevel level, LogSource const &source={})
Creates a new LogContext with the given level and source.
std::function< void(LogLevel logLevel, const std::string &logMessage, const std::string &file, const std::string &method, const int line)> LogPrinter
Definition: Logger.h:251
bool isDebugEnabled(LogModule module) const
Definition: Logger.h:277
void setAllModulesToLogLevel(LogLevel level)
Definition: Logger.h:293
A generic object pool implementation.
Definition: ObjectPool.h:24
A context encapsulating the details of a single log message to be passed to the Logger.
Definition: Logger.h:170
LogContext()=default
Creates a context with an empty message with Info level and no source.
LogContext(LogLevel level, LogSource const &source={})
Creates a context with an empty message with the given level and source.
Definition: Logger.h:180
void init(LogLevel level, LogSource const &source)
Initializes the context with an empty message and the given level and source.
Definition: Logger.h:186
LogContext & operator<<(T const &value)
Appends to the message.
Definition: Logger.h:197
The main namespace for the PcapPlusPlus lib.
Definition: AssertionUtils.h:19
std::string getErrorString(int errnum)
LogModule
An enum representing all PcapPlusPlus modules.
Definition: Logger.h:52
@ PacketLogModuleRawPacket
RawPacket module (Packet++)
Definition: Logger.h:57
@ PacketLogModulePayloadLayer
PayloadLayer module (Packet++)
Definition: Logger.h:65
@ PcapLogModuleXdpDevice
XdpDevice module (Pcap++)
Definition: Logger.h:113
@ PacketLogModuleUdpLayer
UdpLayer module (Packet++)
Definition: Logger.h:67
@ PacketLogModuleTelnetLayer
TelnetLayer module (Packet++)
Definition: Logger.h:94
@ CommonLogModuleTablePrinter
Table printer module (Common++)
Definition: Logger.h:55
@ PacketLogModuleFtpLayer
FtpLayer module (Packet++)
Definition: Logger.h:98
@ PacketLogModuleGreLayer
GreLayer module (Packet++)
Definition: Logger.h:75
@ PacketLogModuleLLCLayer
LLCLayer module (Packet++)
Definition: Logger.h:96
@ PacketLogModuleVlanLayer
VlanLayer module (Packet++)
Definition: Logger.h:68
@ PcapLogModuleKniDevice
KniDevice module (Pcap++)
Definition: Logger.h:112
@ PacketLogModuleSomeIpLayer
SomeIpLayer module (Packet++)
Definition: Logger.h:99
@ PacketLogModuleEthLayer
EthLayer module (Packet++)
Definition: Logger.h:62
@ PacketLogModuleSll2Layer
Sll2Layer module (Packet++)
Definition: Logger.h:78
@ PacketLogModuleSdpLayer
SdpLayer module (Packet++)
Definition: Logger.h:84
@ PacketLogModuleIPv4Layer
IPv4Layer module (Packet++)
Definition: Logger.h:63
@ PacketLogModuleArpLayer
ArpLayer module (Packet++)
Definition: Logger.h:61
@ PacketLogModuleRadiusLayer
RadiusLayer module (Packet++)
Definition: Logger.h:85
@ PacketLogModuleDhcpV6Layer
DhcpV6Layer module (Packet++)
Definition: Logger.h:81
@ PacketLogModuleSipLayer
SipLayer module (Packet++)
Definition: Logger.h:83
@ PacketLogModuleNflogLayer
NflogLayer module (Packet++)
Definition: Logger.h:79
@ PcapLogModuleRemoteDevice
WinPcapRemoteDevice module (Pcap++)
Definition: Logger.h:106
@ PacketLogModuleNdpLayer
NdpLayer module (Packet++)
Definition: Logger.h:97
@ CommonLogModuleIpUtils
IP Utils module (Common++)
Definition: Logger.h:54
@ PacketLogModuleIcmpLayer
IcmpLayer module (Packet++)
Definition: Logger.h:73
@ PacketLogModuleNtpLayer
NtpLayer module (Packet++)
Definition: Logger.h:93
@ PacketLogModuleVrrpLayer
Vrrp Record module (Packet++)
Definition: Logger.h:89
@ PcapLogModuleFileDevice
FileDevice module (Pcap++)
Definition: Logger.h:108
@ PacketLogModuleGtpLayer
GtpLayer module (Packet++)
Definition: Logger.h:86
@ PacketLogModuleSSHLayer
SSHLayer module (Packet++)
Definition: Logger.h:88
@ PacketLogModuleDhcpLayer
DhcpLayer module (Packet++)
Definition: Logger.h:80
@ PcapLogModuleDpdkDevice
DpdkDevice module (Pcap++)
Definition: Logger.h:111
@ PacketLogModuleWakeOnLanLayer
WakeOnLanLayer module (Packet++)
Definition: Logger.h:101
@ PacketLogModulePPPoELayer
PPPoELayer module (Packet++)
Definition: Logger.h:70
@ PacketLogModuleIPReassembly
IPReassembly module (Packet++)
Definition: Logger.h:91
@ PacketLogModuleTcpReassembly
TcpReassembly module (Packet++)
Definition: Logger.h:90
@ PacketLogModuleHttpLayer
HttpLayer module (Packet++)
Definition: Logger.h:69
@ PacketLogModuleWireGuardLayer
WireGuardLayer module (Packet++)
Definition: Logger.h:103
@ PacketLogModuleBgpLayer
GtpLayer module (Packet++)
Definition: Logger.h:87
@ PacketLogModuleSllLayer
SllLayer module (Packet++)
Definition: Logger.h:77
@ CommonLogModuleGenericUtils
Generic Utils (Common++)
Definition: Logger.h:56
@ PacketLogModuleIgmpLayer
IgmpLayer module (Packet++)
Definition: Logger.h:82
@ PacketLogModuleStpLayer
StpLayer module (Packet++)
Definition: Logger.h:95
@ PcapLogModuleWinPcapLiveDevice
WinPcapLiveDevice module (Pcap++)
Definition: Logger.h:105
@ PacketLogModuleIPv6Layer
IPv6Layer module (Packet++)
Definition: Logger.h:64
@ PacketLogModuleMplsLayer
MplsLayer module (Packet++)
Definition: Logger.h:72
@ PacketLogModuleAsn1Codec
Asn1Codec module (Packet++)
Definition: Logger.h:60
@ PcapLogModuleNetworkUtils
Network Utils module (Pcap++)
Definition: Logger.h:114
@ PacketLogModuleDnsLayer
DnsLayer module (Packet++)
Definition: Logger.h:71
@ PacketLogModuleDoIpLayer
DoipLayer module (Packet++)
Definition: Logger.h:104
@ PcapLogModulePfRingDevice
PfRingDevice module (Pcap++)
Definition: Logger.h:109
@ PacketLogModuleSSLLayer
SSLLayer module (Packet++)
Definition: Logger.h:76
@ PacketLogModulePacket
Packet module (Packet++)
Definition: Logger.h:58
@ PacketLogModuleSomeIpSdLayer
SomeIpSdLayer module (Packet++)
Definition: Logger.h:100
@ PcapLogModuleMBufRawPacket
MBufRawPacket module (Pcap++)
Definition: Logger.h:110
@ PacketLogModuleLayer
Layer module (Packet++)
Definition: Logger.h:59
@ PacketLogModuleTcpLayer
TcpLayer module (Packet++)
Definition: Logger.h:66
@ PacketLogModuleIcmpV6Layer
IcmpV6Layer module (Packet++)
Definition: Logger.h:74
@ PcapLogModuleLiveDevice
PcapLiveDevice module (Pcap++)
Definition: Logger.h:107
@ PacketLogModuleSmtpLayer
SmtpLayer module (Packet++)
Definition: Logger.h:102
@ PacketLogModuleIPSecLayer
IPSecLayers module (Packet++)
Definition: Logger.h:92
LogLevel
Definition: Logger.h:149
@ Info
Info level logs and above are emitted.
@ Warn
Warning level logs and above are emitted.
@ Error
Error level logs are emitted.
@ Debug
Debug level logs and above are emitted.
@ Off
No log messages are emitted.
Definition: Logger.h:122
const char * function
Definition: Logger.h:141
LogModule logModule
Definition: Logger.h:143
constexpr LogSource()=default
Default constructor for LogSource.
constexpr LogSource(LogModule logModule)
Definition: Logger.h:128
constexpr LogSource(LogModule logModule, const char *file, const char *function, int line)
Definition: Logger.h:136
const char * file
Definition: Logger.h:140
int line
Definition: Logger.h:142