Skip to content

a log library based on c++11 for linux and mac

License

Notifications You must be signed in to change notification settings

chloro-pn/pnlog

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

92 Commits
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

pnlog

a c++ log library based on c++11 for linux.

features

  • 支持多日志文件同时打开
  • 支持异步/同步模式
  • 支持日志等级调整(非线程安全)
  • 支持基于时间间隔的日志回滚(目前不会删除过期日志)
  • 支持日志文件运行时关闭
  • 支持输出重定向
  • 支持运行时打开/关闭时间戳 (非线程安全)
  • 通过使用log-buf的对象池,避免频繁的申请/释放动态内存。
  • 通过使用自旋锁而不是std::mutex,减少并发开销。

benchmark

build

pnlog uses cmake build tool to build the project.

mkdir build && cd build
cmake ..
make

license

MIT License

how to use

you can research example code in /examples dir to learn how to use pnlog.

  • write to stdout.
  • Global object backend's type is std::shared_ptr, and you can only have one object of the class BackEnd, which means the constructor of class BackEnd is privated.
  • Global object backend is in charge of opening, closing, and reopening various log files, and each of log files managed by backend is identified by an index.
  • stdcout is opened by backend by default and it's index is 0.
  • You can call backend's get_capture method to create a std::shared obejct from backend. The parameter is the index of log file, each log submitd by CapTure object will be appended to log file identified by index.
#include <iostream>
#include "../include/pnlog.h"
#include "../include/file_out_stream.h"

using pnlog::backend;

int main() {
  auto capture = backend->get_capture(0);
  for(int i = 0; i < 10; ++i) {
    if(i >= 5) {
      capture->enable_time();
    }
    if(i >= 8) {
      capture->disable_time();
    }
    if(i % 2 == 0) {
      capture->trace(piece("hello world ! ", " ", i));
    }
    else {
      capture->trace(piece("hello world ! ", " ", i), __FILE__, __LINE__);
    }
  }
  return 0;
}
  • use enable_time() method and disable_time() method to choose timestamp or not.
#include "../include/pnlog.h"

using pnlog::backend;
int main() {
  auto cap1 = backend->get_capture(0);
  cap1->enable_time();
  for(int i = 0; i < 10; ++i) {
    if(i > 5) {
      cap1->setLevel(pnlog::CapTure::Level::PN_WARNING);
    }
    cap1->trace(piece("example cap1 ", i));
  }
  return 0;
}
  • CapTure object can be constructed by invalid index(no log files correspond to it), under the condition ,the log will be abandoned.
  • More than one CapTure object can be constructed by the same index, and each object has its own log_level and format.
#include "../include/pnlog.h"

using pnlog::backend;
int main() {
  auto cap1 = backend->get_capture(0);
  auto cap2 = backend->get_capture(0);
  auto cap3 = backend->get_capture(1); // index 1 is invalid.
  cap1->enable_time();
  for(int i = 0; i < 5; ++i) {
    cap1->trace(piece("example cap1 ", i));
    cap2->trace(piece("example cap2 ", i), __FILE__, __LINE__);
    cap3->trace(piece("example cap3 ")); // do nothing.
  }
  return 0;
}
  • Use backend->open() method to open a log file.
  • Log file can be opened by sync model (which means call out_stream_base-> write() for each log)and async model (store a lot of logs in memory cache and write to log file once).
#include "../include/pnlog.h"
#include "../include/file_out_stream.h"
#include <cstdio>
#include <iostream>

using pnlog::backend;
int main() {
  pnlog::BackEnd::options op;
  op.asyn = false; // op.asyn = true by default.
  backend->open(op, 1, new pnlog::FileOutStream("example3log.txt"));
  auto cap1 = backend->get_capture(1);
  cap1->enable_time();
  for(int i = 0; i < 5; ++i) {
    cap1->trace(piece("example3 ", i));
  }
  cap1->close();
  std::cout << "syn model \n";
  return 0;
}
  • capture's destructor would not close log file identified by index, and the log file would be closed by backend's destructor.However, you can call capture->close() method to do it.
#include "../include/pnlog.h"
#include "../include/file_out_stream.h"
#include <iostream>
#include <cstdio>

using pnlog::backend;
int main() {
  pnlog::BackEnd::options op;
  backend->open(op, 1, new pnlog::FileOutStream("example4log.txt"));
  auto cap1 = backend->get_capture(1);
  cap1->enable_time();
  for(int i = 0; i < 5; ++i) {
    cap1->trace(piece("example cap1 ", i));
  }
  cap1->close();
  std::cout << "the program is running now, but log file has been closed, try to "
               "check it out.\n";
  getchar();
  return 0;
}
  • log rotate can be supported by set BackEnd::options object. Pnlog support duration-based rotate model.
#include "../include/pnlog.h"
#include "../include/file_out_stream.h"
#include <unistd.h>
#include <iostream>

using pnlog::backend;
int main() {
  pnlog::BackEnd::options op;
  //Rotate by duration, and each 3 seconds rotate the log file.
  op.duration_rotating = true;
  op.duration = std::chrono::seconds(3);
  //set rotate file path and name.
  op.path = "example5";
  backend->open(op, 1, new pnlog::FileOutStream("example5.txt"));
  auto cap1 = backend->get_capture(1);
  cap1->enable_time();
  for(int i = 0; i < 10; ++i) {
    cap1->trace(piece("example cap1 ", i));
    sleep(1);
    std::cout << "wait for 9 : "<< i << std::endl;
  }
  return 0;
}
  • thread-safe reopen method and close method.
#include "../include/pnlog.h"
#include "../include/file_out_stream.h"
#include <thread>
#include <iostream>
#include <unistd.h>

using namespace pnlog;

int main() {
  BackEnd::options op;
  backend->open(op, 1, new FileOutStream("example6log.txt"));
  backend->open(op, 2, new FileOutStream("example6_closedlog.txt"));
  auto cap = backend->get_capture(1);
  auto cap2 = backend->get_capture(2);
  std::thread th([&]()->void {
                   for(int i = 0; i < 10000; ++i) {
                     cap->trace(piece("writing in back thread. ", i));
                     cap2->trace(piece("waiting for close.", i));
                   }
                 });
  backend->close(2);
  usleep(1e4);
  backend->reopen(1, new FileOutStream("example6_reopenlog.txt"));
  th.join();
  std::cout << "please check example6*.txt file \n";
  return 0;
}
  • Configure log schedule for your class.
#include "../include/pnlog.h"
#include "../include/file_out_stream.h"
#include <memory>

using pnlog::CapTure;
using pnlog::backend;
using pnlog::BackEnd;

class test {
private:
  int num_;//your data member.
  std::shared_ptr<CapTure> logger;

public:
  test():num_(0), logger(backend->get_capture(0)) {
    logger->enable_time();
    logger->setLevel(CapTure::Level::PN_DEBUG);
  }

  void add() {
    ++num_;
    logger->trace(piece("num change : ", num_), __FILE__, __LINE__);
  }

  void cut_down() {
    if(num_ <= 0) {
      logger->warning(piece("num now : ", num_), __FILE__, __LINE__);
    }
    --num_;
    return;
  }

  ~test() = default;
};

int main() {
  test t;
  t.add();
  t.cut_down();
  t.cut_down();
  return 0;
}

About

a log library based on c++11 for linux and mac

Topics

Resources

License

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published