MagAO-X
Operations Applications Utilities Source
logManager.hpp
Go to the documentation of this file.
1 /** \file logManager.hpp
2  * \brief The MagAO-X log manager.
3  * \author Jared R. Males (jaredmales@gmail.com)
4  *
5  * \ingroup logger_files
6  *
7  * History:
8  * - 2017-06-27 created by JRM
9  */
10 
11 #ifndef logger_logManager_hpp
12 #define logger_logManager_hpp
13 
14 #include <memory>
15 #include <list>
16 
17 #include <thread>
18 
19 #include <mutex>
20 #include <ratio>
21 
22 
23 #include <flatlogs/flatlogs.hpp>
24 using namespace flatlogs;
25 
26 #include "generated/logTypes.hpp"
27 
28 namespace MagAOX
29 {
30 namespace logger
31 {
32 
33 /// The standard MagAOX log manager, used for both process logs and telemetry streams.
34 /** Manages the formatting and queueing of the log entries.
35  *
36  * A log entry is made using one of the standard log types. These are formatted into a binary stream and
37  * stored in a std::list. This occurs in the calling thread. The insertion into the list is mutex-ed, so
38  * it is safe to make logs from different threads concurrently.
39  *
40  * Write-to-disk occurs in a separate thread, which
41  * is normally set to the lowest priority so as not to interfere with higher-priority tasks. The
42  * log thread cycles through pending log entries in the list, dispatching them to the logFile.
43  *
44  * The template parameter logFileT is one of the logFile types, which is used to actually write to disk.
45  *
46  * Example:
47  \code
48  logManager<logFileRaw> log;
49  log.m_logThreadStart();
50 
51  //do stuff
52 
53  log<loop_closed>(); //An empty log, which is used to log discrete events.
54 
55  //more stuff happens
56 
57  log<text_log>("a log entry with a std::string message type");
58 
59  \endcode
60  *
61  * \todo document all the requirements of logFileT
62  *
63  * \tparam logFileT a logFile type with a writeLog method.
64  *
65  * \ingroup logger
66  */
67 template<class logFileT>
68 struct logManager : public logFileT
69 {
70  ///\todo Make these protected members, with appropriate access methods
71  //-->
72  std::list<bufferPtrT> m_logQueue; ///< Log entries are stored here, and writen to the file by the log thread.
73 
74  std::thread m_logThread; ///< A separate thread for actually writing to the file.
75  std::mutex m_qMutex; ///< Mutex for accessing the m_logQueue.
76 
77  bool m_logShutdown {false}; ///< Flag to signal the log thread to shutdown.
78 
79  unsigned long m_writePause {MAGAOX_default_writePause}; ///< Time, in nanoseconds, to pause between successive batch writes to the file. Default is 1e9. Configure with logger.writePause.
80 
81  logPrioT m_logLevel {logPrio::LOG_INFO}; ///< The minimum log level to actually record. Logs with level below this are rejected. Default is INFO. Configure with logger.logLevel.
82 
83  int m_logThreadPrio {0};
84 
85  bool m_logThreadRunning {false};
86  //<--end of todo
87 
88  /// Default c'tor.
89  logManager();
90 
91  /// Destructor.
92  ~logManager();
93 
94  /// Set a new value of writePause
95  /** Updates m_writePause with new value.
96  *
97  * \returns 0 on success
98  * \returns -1 on error (if wp == 0).
99  */
100  int writePause( const unsigned long & wp /**< [in] the new value of writePause */);
101 
102  /// Get the current value of writePause
103  /** \returns the value m_writePause.
104  */
105  unsigned long writePause();
106 
107  /// Set a new value of logLevel
108  /** Updates m_logLevel with new value.
109  * Will return an error and take no actions if the argument
110  * is outside the range of the logLevels enum.
111  *
112  * \returns 0 on success
113  * \returns -1 on error.
114  */
115  int logLevel( logPrioT newLev /**< [in] the new value of logLevel */);
116 
117  /// Get the current value of logLevel
118  /** \returns the value m_logLevel
119  */
120  logPrioT logLevel();
121 
122  /// Set a new value of logThreadPrio
123  /** Updates m_logThreadPrio with new value.
124  * If the argument is < 0, this sets m_logThreadPrio to 0.
125  * Will not set > 98, and returns -1 with no changes in this case.
126  *
127  * \returns 0 on success
128  * \returns -1 on error (> 98).
129  */
130  int logThreadPrio( int newPrio /**< [in] the new value of logThreadPrio */);
131 
132  /// Get the current value of logThreadPrio
133  /** \returns the value m_logThreadPrio
134  */
135  int logThreadPrio();
136 
137  /// Get status of the log thread running flag
138  /** \returns the current value of m_logThreadRunning
139  */
140  bool logThreadRunning();
141 
142  ///Setup an application configurator for the logger section
143  int setupConfig( mx::appConfigurator & config /**< [in] an application configuration to setup */);
144 
145  ///Load the logger section from an application configurator
146  /**
147  */
148  int loadConfig( mx::appConfigurator & config /**< [in] an application configuration from which to load values */);
149 
150 
151  ///Thread starter, called by logThreadStart on thread construction. Calls logThreadExec.
152  static void _logThreadStart( logManager * l /**< [in] a pointer to a logger instance (normally this) */);
153 
154  /// Start the logger thread.
155  int logThreadStart();
156 
157  /// Execute the logger thread.
158  void logThreadExec();
159 
160  /// Create a log formatted log entry, filling in a buffer.
161  /** This is where the timestamp of the log entry is set.
162  *
163  * \tparam logT is a log entry type
164  *
165  * \returns 0 on success, -1 on error.
166  */
167  template<typename logT>
168  static int createLog( bufferPtrT & logBuffer, ///< [out] a shared_ptr<logBuffer>, which will be allocated and populated with the log entry
169  const typename logT::messageT & msg, ///< [in] the message to log (could be of type emptyMessage)
170  const logPrioT & level ///< [in] the level (verbosity) of this log
171  );
172 
173  /// Create a log formatted log entry, filling in a buffer.
174  /** This version has the timestamp provided.
175  *
176  * \tparam logT is a log entry type
177  *
178  * \returns 0 on success, -1 on error.
179  */
180  template<typename logT>
181  static int createLog( bufferPtrT & logBuffer, ///< [out] a shared_ptr<logBuffer>, which will be allocated and populated with the log entry
182  const timespecX & ts, ///< [in] the timestamp of this log entry.
183  const typename logT::messageT & msg, ///< [in] the message to log (could be of type emptyMessage)
184  const logPrioT & level ///< [in] the level (verbosity) of this log
185  );
186 
187  /// Make a log entry, including a message.
188  /**
189  * \tparam logT is a log entry type
190  */
191  template<typename logT>
192  void log( const typename logT::messageT & msg, ///< [in] the message to log
193  logPrioT level = logPrio::LOG_DEFAULT ///< [in] [optional] the log level. The default is used if not specified.
194  );
195 
196  /// Make a log entry, including a message.
197  /**
198  * \tparam logT is a log entry type
199  */
200  template<typename logT>
201  void log( timespecX & ts, ///< [in] the timestamp of the log entry
202  const typename logT::messageT & msg, ///< [in] the message to log
203  logPrioT level = logPrio::LOG_DEFAULT ///< [in] [optional] the log level. The default is used if not specified.
204  );
205 
206  /// Make a log entry with no message.
207  /**
208  * \tparam logT is a log entry type
209  */
210  template<typename logT>
211  void log( logPrioT level = logPrio::LOG_DEFAULT /**< [in] [optional] the log level. The default is used if not specified.*/);
212 
213  /// Make a log entry with no message.
214  /**
215  * \tparam logT is a log entry type
216  */
217  template<typename logT>
218  void log( timespecX & ts, ///< [in] the timestamp of the log entry
219  logPrioT level = logPrio::LOG_DEFAULT ///< [in] [optional] the log level. The default is used if not specified.
220  );
221 
222 };
223 
224 template<class logFileT>
226 {
227 }
228 
229 template<class logFileT>
231 {
232  m_logShutdown = true;
233 
234  if(m_logThread.joinable()) m_logThread.join();
235 
236  //One last check to see if there are any unwritten logs.
237  if( !m_logQueue.empty() ) logThreadExec();
238 
239 }
240 
241 template<class logFileT>
242 int logManager<logFileT>::writePause( const unsigned long & wp )
243 {
244  if(wp == 0) return -1;
245 
246  m_writePause = wp;
247 
248  return 0;
249 }
250 
251 template<class logFileT>
253 {
254  return m_writePause;
255 }
256 
257 template<class logFileT>
259 {
260 
261 
262  m_logLevel = newLev;
263 
264  return 0;
265 }
266 
267 template<class logFileT>
269 {
270  return m_logLevel;
271 }
272 
273 template<class logFileT>
275 {
276  if(newPrio > 98) return -1; //clamped at 98 for safety.
277  if(newPrio < 0) newPrio = 0;
278 
279  m_logThreadPrio = newPrio;
280  return 0;
281 }
282 
283 template<class logFileT>
285 {
286  return m_logThreadPrio;
287 }
288 
289 template<class logFileT>
291 {
292  return m_logThreadRunning;
293 }
294 
295 template<class logFileT>
296 int logManager<logFileT>::setupConfig( mx::appConfigurator & config )
297 {
298  config.add("logger.logDir","L", "logDir",mx::argType::Required, "logger", "logDir", false, "string", "The directory for log files");
299  config.add("logger.logExt","", "logExt",mx::argType::Required, "logger", "logExt", false, "string", "The extension for log files");
300  config.add("logger.maxLogSize","", "maxLogSize",mx::argType::Required, "logger", "maxLogSize", false, "string", "The maximum size of log files");
301  config.add("logger.writePause","", "writePause",mx::argType::Required, "logger", "writePause", false, "unsigned long", "The log thread pause time in ns");
302  config.add("loger.logThreadPrio", "", "logThreadPrio", mx::argType::Required, "logger", "logThreadPrio", false, "int", "The log thread priority");
303  config.add("logger.logLevel","l", "logLevel",mx::argType::Required, "logger", "logLevel", false, "string", "The log level");
304 
305  return 0;
306 }
307 
308 template<class logFileT>
309 int logManager<logFileT>::loadConfig( mx::appConfigurator & config )
310 {
311  //-- logDir
312  std::string tmp;
313  config(tmp, "logger.logDir");
314  if(tmp != "") this->logPath(tmp);
315 
316  //-- logLevel
317  tmp = "";
318  config(tmp, "logger.logLevel");
319  if(tmp != "")
320  {
321  logPrioT lev;
322 
323  lev = logLevelFromString(tmp);
324 
325  if( lev == logPrio::LOG_DEFAULT ) lev = logPrio::LOG_INFO;
326  if( lev == logPrio::LOG_UNKNOWN )
327  {
328  std::cerr << "Unkown log level specified. Using default (INFO)\n";
329  lev = logPrio::LOG_INFO;
330  }
331  logLevel(lev);
332  }
333 
334 
335  //logExt
336  config(this->m_logExt, "logger.logExt");
337 
338  //maxLogSize
339  config(this->m_maxLogSize, "logger.maxLogSize");
340 
341  //writePause
342  config(m_writePause, "logger.writePause");
343 
344  //logThreadPrio
345  config(m_logThreadPrio, "logger.logThreadPrio");
346 
347  return 0;
348 }
349 
350 template<class logFileT>
352 {
353  l->logThreadExec();
354 }
355 
356 template<class logFileT>
358 {
359  try
360  {
361  m_logThread = std::thread( _logThreadStart, this);
362  }
363  catch( const std::exception & e )
364  {
365  log<software_error>({__FILE__,__LINE__, 0, 0, std::string("Exception on log thread start: ") + e.what()});
366  return -1;
367  }
368  catch( ... )
369  {
370  log<software_error>({__FILE__,__LINE__, 0, 0, "Unkown exception on log thread start"});
371  return -1;
372  }
373 
374  if(!m_logThread.joinable())
375  {
376  log<software_error>({__FILE__, __LINE__, 0, 0, "Log thread did not start"});
377  return -1;
378  }
379 
380  //Always set the m_logThread to lowest priority
381  sched_param sp;
382  sp.sched_priority = m_logThreadPrio;
383 
384  int rv = pthread_setschedparam( m_logThread.native_handle(), SCHED_OTHER, &sp);
385 
386  if(rv != 0)
387  {
388  log<software_error>({__FILE__, __LINE__, 0, rv, std::string("Error setting thread params: ") + strerror(rv)});
389  return -1;
390  }
391 
392  return 0;
393 
394 }
395 
396 template<class logFileT>
398 {
399 
400  m_logThreadRunning = true;
401 
402  std::unique_lock<std::mutex> lock(m_qMutex, std::defer_lock);
403 
404  while(!m_logShutdown || !m_logQueue.empty())
405  {
406  std::list<bufferPtrT>::iterator beg, it, er, end;
407 
408  //Get begin and end once, so we only process logs in the list at this point.
409  //Do it locked so we don't have any collisions with new logs (maybe not necessary)
410  lock.lock();
411  beg = m_logQueue.begin();
412  end = m_logQueue.end();
413  lock.unlock();
414 
415  //Note: we're avoiding use of size() since that could be altered concurrently by a push_back.
416  if( beg != end )
417  {
418  it = beg;
419  while( it != end )
420  {
421  //m_logFile.
422  if( this->writeLog( *it ) < 0)
423  {
424  m_logThreadRunning = false;
425  return;
426  }
427 
428  er = it;
429  ++it;
430 
431  //Erase while locked so we don't collide with a push_back.
432  lock.lock();
433  m_logQueue.erase(er);
434  lock.unlock();
435  }
436  }
437 
438  //m_logFile.
439  this->flush();
440 
441  //We only pause if there's nothing to do.
442  if(m_logQueue.empty() && !m_logShutdown) std::this_thread::sleep_for( std::chrono::duration<unsigned long, std::nano>(m_writePause));
443  }
444 
445  m_logThreadRunning = false;
446 }
447 
448 template<class logFileT>
449 template<typename logT>
451  const typename logT::messageT & msg,
452  const logPrioT & level
453  )
454 {
455  //Very first step is to get the current time.
456  timespecX ts;
457  ts.gettime();
458 
459  return logHeader::createLog<logT>(logBuffer, ts, msg, level);
460 }
461 
462 template<class logFileT>
463 template<typename logT>
465  const timespecX & ts,
466  const typename logT::messageT & msg,
467  const logPrioT & level
468  )
469 {
470  return logHeader::createLog<logT>(logBuffer, ts, msg, level);
471 }
472 
473 template<class logFileT>
474 template<typename logT>
475 void logManager<logFileT>::log( const typename logT::messageT & msg,
476  logPrioT level
477  )
478 {
479  //Step 0 check level.
480  if(level == logPrio::LOG_DEFAULT) level = logT::defaultLevel;
481 
482  if(level > m_logLevel) return; // We do nothing with this.
483 
484  //Step 1 create log
485  bufferPtrT logBuffer;
486  createLog<logT>(logBuffer, msg, level);
487 
488  //Step 2 add log to queue
489  std::lock_guard<std::mutex> guard(m_qMutex); //Lock the mutex before pushing back.
490  m_logQueue.push_back(logBuffer);
491 
492 
493 }
494 
495 template<class logFileT>
496 template<typename logT>
498  const typename logT::messageT & msg,
499  logPrioT level
500  )
501 {
502  //Step 0 check level.
503  if(level == logPrio::LOG_DEFAULT) level = logT::defaultLevel;
504 
505  if(level > m_logLevel) return; // We do nothing with this.
506 
507  //Step 1 create log
508  bufferPtrT logBuffer;
509  createLog<logT>(logBuffer, ts, msg, level);
510 
511  //Step 2 add log to queue
512  std::lock_guard<std::mutex> guard(m_qMutex); //Lock the mutex before pushing back.
513  m_logQueue.push_back(logBuffer);
514 
515 }
516 
517 template<class logFileT>
518 template<typename logT>
520 {
521  log<logT>( emptyMessage(), level );
522 }
523 
524 template<class logFileT>
525 template<typename logT>
527  logPrioT level
528  )
529 {
530  log<logT>( ts, emptyMessage(), level );
531 }
532 
533 
534 } //namespace logger
535 } //namespace MagAOX
536 
537 #endif //logger_logger_hpp
std::thread m_logThread
A separate thread for actually writing to the file.
Definition: logManager.hpp:74
static constexpr logPrioT LOG_INFO
Informational. The info log level is the lowest level recorded during normal operations.
Definition: logPriority.hpp:49
The standard MagAOX log manager, used for both process logs and telemetry streams.
Definition: logManager.hpp:68
std::list< bufferPtrT > m_logQueue
Log entries are stored here, and writen to the file by the log thread.
Definition: logManager.hpp:72
#define MAGAOX_default_writePause
The default logger writePause.
Definition: defaults.hpp:30
logPrioT logLevelFromString(const std::string &str)
Get the log priority from a string, which might have the number or the name.
Empty type for resolving logs with no message.
Definition: empty_log.hpp:24
static constexpr logPrioT LOG_UNKNOWN
Used to denote an unkown log type for internal error handling.
Definition: logPriority.hpp:61
int8_t logPrioT
The type of the log priority code.
Definition: logDefs.hpp:19
void logThreadExec()
Execute the logger thread.
Definition: logManager.hpp:397
void gettime()
Fill the the timespecX with the current time.
Definition: timespecX.hpp:76
std::shared_ptr< char > bufferPtrT
The log entry buffer smart pointer.
Definition: logHeader.hpp:57
std::mutex m_qMutex
Mutex for accessing the m_logQueue.
Definition: logManager.hpp:75
Flatlogs single include file.
static constexpr logPrioT LOG_DEFAULT
Used to denote "use the default level for this log type".
Definition: logPriority.hpp:58
A fixed-width timespec structure.
Definition: timespecX.hpp:31