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  return m_LastError;
315  }
316 
319  {
320  m_LogsEnabled = false;
321  }
322 
324  void enableLogs()
325  {
326  m_LogsEnabled = true;
327  }
328 
331  bool logsEnabled() const
332  {
333  return m_LogsEnabled;
334  }
335 
343  void useContextPooling(bool enabled, std::size_t preallocate = 2, std::size_t maxPoolSize = 10)
344  {
345  m_UseContextPooling = enabled;
346 
347  if (m_UseContextPooling)
348  {
349  m_LogContextPool.setMaxSize(maxPoolSize);
350 
351  if (preallocate > 0)
352  {
353  m_LogContextPool.preallocate(preallocate);
354  }
355  }
356  else
357  {
358  // Clear the pool if we're disabling pooling.
359  m_LogContextPool.clear();
360  }
361  }
362 
366  static Logger& getInstance()
367  {
368  static Logger instance;
369  return instance;
370  }
371 
374  std::unique_ptr<internal::LogContext> createLogContext();
375 
380  std::unique_ptr<internal::LogContext> createLogContext(LogLevel level, LogSource const& source = {});
381 
386  void emit(LogSource const& source, LogLevel level, std::string const& message);
387 
390  void emit(std::unique_ptr<internal::LogContext> message);
391 
392  private:
393  bool m_LogsEnabled;
394  std::array<LogLevel, NumOfLogModules> m_LogModulesArray{};
395  LogPrinter m_LogPrinter;
396 
397  static thread_local std::string m_LastError;
398 
399  bool m_UseContextPooling = true;
400  // Keep a maximum of 10 LogContext objects in the pool.
401  internal::DynamicObjectPool<internal::LogContext> m_LogContextPool{ 10, 2 };
402 
403  // private c'tor - this class is a singleton
404  Logger();
405  };
406 
407 } // namespace pcpp
408 
409 #define PCPP_LOG(level, message) \
410  do \
411  { \
412  auto& logger = pcpp::Logger::getInstance(); \
413  if (logger.shouldLog(level, LOG_MODULE)) \
414  { \
415  auto ctx = \
416  logger.createLogContext(level, pcpp::LogSource(LOG_MODULE, PCAPPP_FILENAME, __FUNCTION__, __LINE__)); \
417  (*ctx) << message; \
418  logger.emit(std::move(ctx)); \
419  } \
420  } while (0)
421 
422 #if PCPP_ACTIVE_LOG_LEVEL >= PCPP_LOG_LEVEL_DEBUG
423 # define PCPP_LOG_DEBUG(message) PCPP_LOG(pcpp::LogLevel::Debug, message)
424 #else
425 # define PCPP_LOG_DEBUG(message) (void)0
426 #endif
427 
428 #if PCPP_ACTIVE_LOG_LEVEL >= PCPP_LOG_LEVEL_WARN
429 # define PCPP_LOG_WARN(message) PCPP_LOG(pcpp::LogLevel::Warn, message)
430 #else
431 # define PCPP_LOG_WARN(message) (void)0
432 #endif
433 
434 #if PCPP_ACTIVE_LOG_LEVEL >= PCPP_LOG_LEVEL_INFO
435 # define PCPP_LOG_INFO(message) PCPP_LOG(pcpp::LogLevel::Info, message)
436 #else
437 # define PCPP_LOG_INFO(message) (void)0
438 #endif
439 
440 #if PCPP_ACTIVE_LOG_LEVEL >= PCPP_LOG_LEVEL_ERROR
441 # define PCPP_LOG_ERROR(message) PCPP_LOG(pcpp::LogLevel::Error, message)
442 #else
443 # define PCPP_LOG_ERROR(message) (void)0
444 #endif
Definition: Logger.h:228
void setLogPrinter(LogPrinter printer)
Definition: Logger.h:303
void enableLogs()
Enable logs in all PcapPlusPlus modules.
Definition: Logger.h:324
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:343
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:331
void resetLogPrinter()
Set the log printer back to the default printer.
static Logger & getInstance()
Definition: Logger.h:366
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:318
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