2014-08-07 03:52:16 +00:00
# G3log : Asynchronous logger with Dynamic Sinks
2014-10-10 00:49:20 -06:00
2014-08-07 03:52:16 +00:00
## EXAMPLE USAGE
#### Optional to use either streaming or printf-like syntax
```
LOG(INFO) < < "streaming API is as easy as ABC or " < < 123 ;
LOGF(WARNING, "Printf-style syntax is also %s", "available");
```
2020-06-06 15:39:27 -07:00
# Content
* [What G3log is ](#what-g3log-is )
* [Conditional Logging ](#conditional-logging )
* [Design by Contract ](#design-by-contract )
* [Detailed API documentation ](API.markdown )
* [Benefits with g3log ](#benefits-with-g3log )
* [G3log with sinks ](#g3log-with-sinks )
* [Crazy simple to create a custom sink ](#crazy-simple )
* [Code Examples ](#code-examples )
* [Custom Sink Walkthrough ](#custom-sink )
* [Default File Logger ](#default-file-logger )
* [Building G3log ](#building-g3log )
* [Prerequisites ](#prerequisites )
* [Configuration Options ](#configuration )
* [Build Commands ](#build-commands )
* [Installation ](#installation )
* [Packaging ](#packaging )
* [Testing ](#testing )
* [CMake Module ](#cmake-module )
* Overview of the [API description ](#overview-api-description )
* [Performance ](#performance )
* [Feedback ](#feedback )
* [Say Thanks ](#say-thanks )
## <a name="what-g3log-is">What G3Log is</a>
2014-08-07 03:52:16 +00:00
2020-06-06 15:39:27 -07:00
* ***G3log*** is the acting name for the third version of g2log and it stands for **g3log with dynamic sinks**
* G3log is an asynchronous, "crash-safe" logger. You can read more about it here [[g2log version]](
http://www.codeproject.com/Articles/288827/g2log-An-efficient-asynchronous-logger-using-Cplus)
* You can choose to use the default log receiver which saves all LOG calls to file, **or** you can choose to use your own custom made log receiver(s), **or** both, **or** as many sinks as you need.
2014-10-10 00:22:32 -06:00
2020-06-06 15:39:27 -07:00
#### <a name="#conditional-logging">Conditional logging</a>
2014-08-07 03:52:16 +00:00
int less = 1; int more = 2
LOG_IF(INFO, (less< more ) ) < < " If [ true ] , then this text will be logged " ;
2020-06-06 15:39:27 -07:00
2014-08-07 03:52:16 +00:00
// or with printf-like syntax
LOGF_IF(INFO, (less< more ) , " if % d < % d then this text will be logged " , less , more ) ;
2014-10-10 00:10:06 -06:00
2020-06-06 15:39:27 -07:00
#### <a name="design-by-contract">Design-by-Contract</a>
*CHECK(false)* will trigger a "fatal" message. It will be logged, and then the
2014-08-07 03:52:16 +00:00
application will exit.
```
CHECK(less != more); // not FATAL
CHECK(less > more) < < "CHECK(false) triggers a FATAL message";
```
2020-06-06 15:39:27 -07:00
2014-08-07 03:52:16 +00:00
2015-11-23 23:17:47 -07:00
## Detailed API documentation
2015-11-23 23:18:27 -07:00
Please look at [API.markdown ](API.markdown ) for detailed API documentation
2014-08-07 03:52:16 +00:00
2014-10-10 00:01:58 -06:00
2020-06-06 15:39:27 -07:00
## <a name="benefits-with-g3log">Benefits you get when using G3log</a>
1. Easy to use, clean syntax and a blazing fast logger.
2014-08-07 03:52:16 +00:00
2. All the slow log I/O disk access is done in a background thread. This ensures that the LOG caller can immediately continue with other tasks and do not have to wait for the LOG call to finish.
3. G3log provides logging, Design-by-Contract [#CHECK ], and flush of log to file at
shutdown. Buffered logs will be written to the sink before the application shuts down.
2020-06-06 15:39:27 -07:00
4. It is thread safe, so using it from multiple threads is completely fine.
2014-08-07 03:52:16 +00:00
2020-06-06 15:39:27 -07:00
5. It is *CRASH SAFE* . It will save the made logs to the sink before it shuts down.
2015-03-02 15:24:35 +00:00
The logger will catch certain fatal events *(Linux/OSX: signals, Windows: fatal OS exceptions and signals)* , so if your application crashes due to, say a segmentation fault, *SIGSEGV* , it will log and save the crash and all previously buffered log entries before exiting.
2014-08-07 03:52:16 +00:00
2020-06-06 15:39:27 -07:00
2014-08-07 03:52:16 +00:00
6. It is cross platform. Tested and used by me or by clients on OSX, Windows, Ubuntu, CentOS
2017-04-02 15:22:35 -07:00
7. G3log and G2log are used worldwide in commercial products as well as hobby projects. G2log is used since early 2011.
2014-08-07 03:52:16 +00:00
2015-03-02 15:25:31 +00:00
8. The code is given for free as public domain. This gives the option to change, use, and do whatever with it, no strings attached.
2020-06-06 15:39:27 -07:00
9. Two versions of g3log exists.
* This version: *[g3log](https://github.com/KjellKod/g3log)* : which is made to facilitate easy adding of custom log receivers. Its tested on at least the following platforms with Linux(Clang/gcc), Windows (mingw, visual studio 2013). My recommendation is to go with g3log if you have full C++14 support (C++11 support up to version: https://github.com/KjellKod/g3log/releases/tag/1.3.1).
* *[g2log ](https://bitbucket.org/KjellKod/g2log )*: The original. Simple, easy to modify and with the most OS support. Clients use g2log on environments such as OSX/Clang, Ubuntu, CentOS, Windows/mingw, Windows/Visual Studio. The focus on g2log is "slow to change" and compiler support. Only well, time tested, features from g3log will make it into g2log. Currently there is not active development or support on g2log but feel free to shoot me a question if you need assistance.
2014-08-07 03:52:16 +00:00
2020-06-06 15:39:27 -07:00
# <a name="g3log-with-sinks">G3log with sinks</a>
[Sinks ](http://en.wikipedia.org/wiki/Sink_(computing )) are receivers of LOG calls. G3log comes with a default sink (*the same as G3log uses*) that can be used to save log to file. A sink can be of *any* class type without restrictions as long as it can either receive a LOG message as a *std::string* **or** as a *g3::LogMessageMover* .
2014-08-07 03:52:16 +00:00
2015-07-19 23:10:56 -06:00
The *std::string* comes pre-formatted. The *g3::LogMessageMover* is a wrapped struct that contains the raw data for custom handling in your own sink.
2014-08-07 03:52:16 +00:00
2020-06-06 15:39:27 -07:00
A sink is *owned* by the G3log and is added to the logger inside a ```std::unique_ptr` ``. The sink can be called though its public API through a *handler* which will asynchronously forward the call to the receiving sink.
2015-09-21 15:51:23 -06:00
2020-06-06 15:39:27 -07:00
It is < a name = "crazy-simple" > crazy simple to create a custom sink< / a > . This example show what is needed to make a custom sink that is using custom log formatting but only using that
2015-09-21 15:51:23 -06:00
for adding color to the default log formatting. The sink forwards the colored log to cout
```cpp
// in file Customsink.hpp
#pragma once
#include <string>
#include <iostream>
#include <g3log/logmessage.hpp>
struct CustomSink {
// Linux xterm color
// http://stackoverflow.com/questions/2616906/how-do-i-output-coloured-text-to-a-linux-terminal
enum FG_Color {YELLOW = 33, RED = 31, GREEN=32, WHITE = 97};
FG_Color GetColor(const LEVELS level) const {
if (level.value == WARNING.value) { return YELLOW; }
if (level.value == DEBUG.value) { return GREEN; }
if (g3::internal::wasFatal(level)) { return RED; }
return WHITE;
}
2020-06-06 15:39:27 -07:00
2015-09-21 15:51:23 -06:00
void ReceiveLogMessage(g3::LogMessageMover logEntry) {
auto level = logEntry.get()._level;
auto color = GetColor(level);
2020-06-06 15:39:27 -07:00
std::cout << "\033[" < < color << " m "
2015-09-21 15:51:23 -06:00
< < logEntry.get (). toString () << " \033 [ m " << std::endl ;
}
};
// in main.cpp, main() function
2018-02-20 22:02:19 -07:00
auto sinkHandle = logworker->addSink(std::make_unique< CustomSink > (),
2014-08-07 03:52:16 +00:00
&CustomSink::ReceiveLogMessage);
2020-06-06 15:39:27 -07:00
2014-08-07 03:52:16 +00:00
```
2020-06-19 13:49:30 -06:00
# Adding and Removing Sinks
You can safely remove and add sinks during the running of your program.
**Keep in mind**
- *Initialization of the logger should happen before you have started any other threads that may call the logger.*
- *Destruction of the logger (RAII concept) should happen AFTER shutdown of other threads that are calling the logger.*
**Adding Sinks**
```cpp
auto sinkHandle1 = logworker->addSink(std::make_unique< CustomSink > (),
&CustomSink::ReceiveLogMessage);
auto sinkHandle2 = logworker->addDefaultLogger(argv[0],
path_to_log_file);
logworker->removeSink(std::move(sinkHandle1)); // this will in a thread-safe manner remove the sinkHandle1
logworker->removeAllSinks(); // this will in a thread-safe manner remove any sinks.
```
2015-09-21 15:51:23 -06:00
**More sinks** can be found in the repository ** [github.com/KjellKod/g3sinks ](https://github.com/KjellKod/g3sinks )**.
2020-06-06 15:39:27 -07:00
# <a name="code-examples">Code Examples</a>
Example usage where a < a name = "custom-sink" > custom sink< / a > is added. A function is called though the sink handler to the actual sink object.
2020-05-31 21:50:15 +01:00
```cpp
2014-08-07 03:52:16 +00:00
// main.cpp
2015-07-19 23:10:56 -06:00
#include <g3log/g3log.hpp>
2015-08-29 15:13:08 +05:00
#include <g3log/logworker.hpp>
2018-02-20 22:02:19 -07:00
#include <memory>
2014-08-07 03:52:16 +00:00
#include "CustomSink.h"
int main(int argc, char**argv) {
2015-07-19 23:10:56 -06:00
using namespace g3;
2015-08-19 10:08:41 -06:00
std::unique_ptr< LogWorker > logworker{ LogWorker::createLogWorker() };
2018-02-20 22:02:19 -07:00
auto sinkHandle = logworker->addSink(std::make_unique< CustomSink > (),
2014-08-07 03:52:16 +00:00
&CustomSink::ReceiveLogMessage);
2020-06-06 15:39:27 -07:00
2014-08-07 03:52:16 +00:00
// initialize the logger before it can receive LOG calls
initializeLogging(logworker.get());
LOG(WARNING) < < "This log call, may or may not happend before"
< < "the sinkHandle->call below";
2020-06-06 15:39:27 -07:00
2014-08-07 03:52:16 +00:00
// You can call in a thread safe manner public functions on your sink
// The call is asynchronously executed on your custom sink.
2020-06-06 15:39:27 -07:00
std::future< void > received = sinkHandle->call(& CustomSink::Foo,
2014-08-07 03:52:16 +00:00
param1, param2);
2020-06-06 15:39:27 -07:00
2020-05-31 21:50:15 +01:00
// If the LogWorker is initialized then at scope exit the g3::internal::shutDownLogging() will be called.
2014-08-07 03:52:16 +00:00
// This is important since it protects from LOG calls from static or other entities that will go out of
2020-06-06 15:39:27 -07:00
// scope at a later time.
2014-08-07 03:52:16 +00:00
//
// It can also be called manually:
2020-05-31 21:50:15 +01:00
g3::internal::shutDownLogging();
2014-08-07 03:52:16 +00:00
}
// some_file.cpp : To show how easy it is to get the logger to work
// in other parts of your software
2015-07-19 23:10:56 -06:00
#include <g3log/g3log.hpp>
2014-08-07 03:52:16 +00:00
void SomeFunction() {
...
LOG(INFO) < < "Hello World";
}
```
2020-06-06 15:39:27 -07:00
Example usage where a the < a name = "default-file-logger" > default file logger</ a > is used **and** a custom sink is added
2020-05-31 21:50:15 +01:00
```cpp
2014-08-07 03:52:16 +00:00
// main.cpp
2015-07-19 23:10:56 -06:00
#include <g3log/g3log.hpp>
#include <g3log/logworker.hpp>
2018-02-20 22:02:19 -07:00
#include <memory>
2014-08-07 03:52:16 +00:00
#include "CustomSink.h"
int main(int argc, char**argv) {
2015-07-19 23:10:56 -06:00
using namespace g3;
2015-08-19 10:08:41 -06:00
auto worker = LogWorker::createLogWorker();
2020-06-06 15:39:27 -07:00
auto defaultHandler = worker->addDefaultLogger(argv[0],
2014-08-07 03:52:16 +00:00
path_to_log_file);
2020-06-06 15:39:27 -07:00
2014-08-07 03:52:16 +00:00
// logger is initialized
2015-09-06 15:35:03 +05:00
g3::initializeLogging(worker.get());
2020-06-06 15:39:27 -07:00
2014-08-07 03:52:16 +00:00
LOG(DEBUG) < < "Make log call, then add another sink";
2020-06-06 15:39:27 -07:00
2018-02-20 22:02:19 -07:00
worker->addSink(std::make_unique< CustomSink > (),
2014-08-07 03:52:16 +00:00
&CustomSink::ReceiveLogMessage);
2020-06-06 15:39:27 -07:00
2014-08-07 03:52:16 +00:00
...
}
```
2020-06-06 15:39:27 -07:00
# <a name="building-g3log">Building G3log</a>
2014-08-07 03:52:16 +00:00
2019-05-14 12:42:15 +08:00
```
git clone https://github.com/KjellKod/g3log
cd g3log
mkdir build
cd build
```
2020-06-06 15:39:27 -07:00
## <a name="prerequisites">Prerequisites</a>
2019-05-14 12:42:15 +08:00
Assume you have got your shiny C++14 compiler installed, you also need these tools to build g3log from source:
- CMake (*Required*)
g3log uses CMake as a one-stop solution for configuring, building, installing, packaging and testing on Windows, Linux and OSX.
- Git (*Optional but Recommended*)
2020-06-06 15:39:27 -07:00
When building g3log it uses git to calculate the software version from the commit history of this repository. If you don't want that, or your setup does not have access to git, or you download g3log source archive from the GitHub Releases page so that you do not have the commit history downloaded, you can instead pass in the version as part of the CMake build arguments. See this [_issue_ ](https://github.com/KjellKod/g3log/issues/311#issuecomment-488829282 ) for more information.
2019-05-14 12:42:15 +08:00
```
cmake -DVERSION=1.3.2 ..
```
2020-06-06 15:39:27 -07:00
## <a name="configuration">Configuration Options</a>
2019-05-14 12:42:15 +08:00
g3log provides following CMake options (and default values):
```
$ cmake -LAH # List non-advanced cached variables. See `cmake --help` for more details.
...
// Fatal (fatal-crashes/contract) examples
ADD_FATAL_EXAMPLE:BOOL=ON
// g3log performance test
ADD_G3LOG_BENCH_PERFORMANCE:BOOL=OFF
// g3log unit tests
ADD_G3LOG_UNIT_TEST:BOOL=OFF
// Use DBUG logging level instead of DEBUG.
// By default DEBUG is the debugging level
CHANGE_G3LOG_DEBUG_TO_DBUG:BOOL=OFF
// Specifies the build type on single-configuration generators.
2020-06-06 15:39:27 -07:00
// Possible values are empty, Debug, Release, RelWithDebInfo, MinSizeRel, …
2019-05-14 12:42:15 +08:00
CMAKE_BUILD_TYPE:STRING=
// Install path prefix, prepended onto install directories.
// This variable defaults to /usr/local on UNIX
// and c:/Program Files/${PROJECT_NAME} on Windows.
CMAKE_INSTALL_PREFIX:PATH=
// The prefix used in the built package.
// On Linux, if this option is not set:
// 1) If CMAKE_INSTALL_PREFIX is given, then it will be
// set with the value of CMAKE_INSTALL_PREFIX by g3log.
2020-06-06 15:39:27 -07:00
// 2) Otherwise, it will be set as /usr/local by g3log.
2019-05-14 12:42:15 +08:00
CPACK_PACKAGING_INSTALL_PREFIX:PATH=
2014-08-07 03:52:16 +00:00
2019-05-14 12:42:15 +08:00
// Enable Visual Studio break point when receiving a fatal exception.
// In __DEBUG mode only
DEBUG_BREAK_AT_FATAL_SIGNAL:BOOL=OFF
// Vectored exception / crash handling with improved stack trace
ENABLE_FATAL_SIGNALHANDLING:BOOL=ON
// Vectored exception / crash handling with improved stack trace
ENABLE_VECTORED_EXCEPTIONHANDLING:BOOL=ON
// iOS version of library.
G3_IOS_LIB:BOOL=OFF
// Log full filename
G3_LOG_FULL_FILENAME:BOOL=OFF
// Build shared library
G3_SHARED_LIB:BOOL=ON
// Build shared runtime library MSVC
G3_SHARED_RUNTIME:BOOL=ON
// Turn ON/OFF log levels.
// An disabled level will not push logs of that level to the sink.
// By default dynamic logging is disabled
USE_DYNAMIC_LOGGING_LEVELS:BOOL=OFF
// Use dynamic memory for message buffer during log capturing
USE_G3_DYNAMIC_MAX_MESSAGE_SIZE:BOOL=OFF
...
2019-05-02 15:32:13 -06:00
```
2019-05-14 12:42:15 +08:00
For additional option context and comments please also see [Options.cmake ](https://github.com/KjellKod/g3log/blob/master/Options.cmake )
If you want to leave everything as it was, then you should:
2019-05-02 15:32:13 -06:00
```
2019-05-14 12:42:15 +08:00
cmake ..
```
You may also specify one or more of those options listed above from the command line.
For example, on Windows:
```
cmake .. -G "Visual Studio 15 2017"
-DG3_SHARED_LIB=OFF
-DCMAKE_INSTALL_PREFIX=C:/g3log
-DADD_G3LOG_UNIT_TEST=ON
-DADD_FATAL_EXAMPLE=OFF
```
will use a Visual Studio 2017 solution generator, build g3log as a static library, headers and libraries will be installed to `C:\g3log` when installed from source, enable unit testing, but do not build fatal example.
2014-08-07 03:52:16 +00:00
2019-05-14 12:42:15 +08:00
MinGW users on Windows may find they should use a different generator:
2018-01-09 07:22:08 +03:00
```
2019-05-14 12:42:15 +08:00
cmake .. -G "MinGW Makefiles"
2018-01-09 07:22:08 +03:00
```
2019-05-14 12:42:15 +08:00
By default, headers and libraries will be installed to `/usr/local` on Linux when installed from build tree via `make install` . You may overwrite it by:
```
cmake .. -DCMAKE_INSTALL_PREFIX=/usr
```
This will install g3log to `/usr` instead of `/usr/local` .
2014-08-07 03:52:16 +00:00
2019-05-14 12:42:15 +08:00
Linux/OSX package maintainers may be interested in the `CPACK_PACKAGING_INSTALL_PREFIX` . For example:
2014-08-07 03:52:16 +00:00
```
2019-05-14 12:42:15 +08:00
cmake .. -DCPACK_PACKAGING_INSTALL_PREFIX=/usr/local
2014-08-07 03:52:16 +00:00
```
2019-05-14 12:42:15 +08:00
2020-06-06 15:39:27 -07:00
## <a name="build-commands">Build Commands</a>
2019-05-14 12:42:15 +08:00
Once the configuration is done, you may build g3log with:
```
# Suppose you are still in the `build` directory. I won't repeat it anymore!
cmake --build . --config Release
```
You may also build it with a system-specific way.
On Linux, OSX and MinGW:
```
make
```
On Windows:
```
msbuild g3log.sln /p:Configuration=Release
```
Windows users can also open the generated Visual Studio solution file and build it happily.
2020-06-06 15:39:27 -07:00
## <a name="installing">Installation</a>
2019-05-14 12:42:15 +08:00
Install from source in a CMake way:
```
cmake --build . --target install
```
Linux users may also use:
2017-03-24 13:49:24 -06:00
```
sudo make install
```
2019-05-14 12:42:15 +08:00
You may also create a package first and install g3log with it. See the next section.
2020-06-06 15:39:27 -07:00
## <a name=packaging>Packaging</a>
2019-05-14 12:42:15 +08:00
A CMake way:
```
cmake --build . --config Release --target package
```
or
```
cpack -C Release
```
if the whole library has been built in the previous step.
It will generate a ZIP package on Windows, and a DEB package on Linux.
Linux users may also use a Linux way:
2017-03-24 13:49:24 -06:00
```
make package
```
2014-08-07 03:52:16 +00:00
2019-05-14 12:42:15 +08:00
If you want to use a different package generator, you should specify a `-G` option.
On Windows:
2018-01-09 07:22:08 +03:00
```
2019-05-14 12:42:15 +08:00
cpack -C Release -G NSIS;7Z
2018-01-09 07:22:08 +03:00
```
2019-05-14 12:42:15 +08:00
this will create a installable NSIS package and a 7z package.
*Note:* To use the NSIS generator, you should install [```NSIS``` ](https://nsis.sourceforge.io/Download ) first.
On Linux:
2018-01-09 07:22:08 +03:00
```
2019-05-14 12:42:15 +08:00
cpack -C Release -G TGZ
2018-01-09 07:22:08 +03:00
```
2019-05-14 12:42:15 +08:00
this will create a .tar.gz archive for you.
Once done, you may install or uncompress the package file to the target machine. For example, on Debian or Ubuntu:
2018-01-09 07:22:08 +03:00
```
2019-05-14 12:42:15 +08:00
sudo dpkg -i g3log-< version > -Linux.deb
```
will install the g3log library to `CPACK_PACKAGING_INSTALL_PREFIX` .
2020-06-06 15:39:27 -07:00
## <a name="testing">Testing</a>
2019-05-14 12:42:15 +08:00
2020-06-06 15:39:27 -07:00
By default, tests will not be built. To enable unit testing, you should turn on `ADD_G3LOG_UNIT_TEST` .
2019-05-14 12:42:15 +08:00
Suppose the build process has completed, then you can run the tests with:
```
2019-12-14 21:51:10 +08:00
ctest -C Release
2019-05-14 12:42:15 +08:00
```
or:
```
make test
2018-01-09 07:22:08 +03:00
```
2019-05-14 12:42:15 +08:00
for Linux users.
2020-06-06 15:39:27 -07:00
or for a detailed gtest output of all the tests:
```
cd build;
../scripts/runAllTests.sh
```
2019-05-14 12:42:15 +08:00
2020-06-06 15:39:27 -07:00
## <a name="cmake-module">CMake module</a>
2019-05-14 12:42:15 +08:00
2020-07-08 22:42:55 -06:00
g3log comes with a CMake module. Once installed, it can be found under `${CMAKE_INSTALL_PREFIX}/lib/cmake/g3log` . Users can use g3log in a CMake-based project this way:
2018-01-09 07:22:08 +03:00
2014-08-07 03:52:16 +00:00
```
2020-07-08 22:42:55 -06:00
find_package(g3log CONFIG REQUIRED)
target_link_libraries(main PRIVATE g3log)
2014-08-07 03:52:16 +00:00
```
2020-07-08 22:42:55 -06:00
To make sure that CMake can find g3log, you also need to tell CMake where to search for it:
2014-08-07 03:52:16 +00:00
```
2019-05-14 12:42:15 +08:00
cmake .. -DCMAKE_PREFIX_PATH=< g3log ' s install prefix >
2014-08-07 03:52:16 +00:00
```
2020-06-06 15:39:27 -07:00
# Overview of the <a name="overview-api-description">API description</a>
Most of the API that you need for using g3log is described in this readme. For more API documentation and examples please continue to read the [API readme ](API.markdown ). Examples of what you will find here are:
2015-09-21 16:23:27 -06:00
* Sink creation and utilization
2020-06-06 15:39:27 -07:00
* Logging levels
2015-09-21 16:23:27 -06:00
* disable/enabled levels at runtime
* custom logging levels
* Fatal handling
* custom fatal handling
* pre fatal hook
* override of signal handling
* disable fatal handling
* LOG calls
* CHECK calls
2015-09-21 15:51:23 -06:00
2020-06-06 15:39:27 -07:00
# <a name="performance">Performance</a>
G3log aims to keep all background logging to sinks with as little log overhead as possible to the logging sink and with as small "worst case latency" as possible. For this reason g3log is a good logger for many systems that deal with critical tasks. Depending on platform the average logging overhead will differ. On my 2010 laptop the average call, when doing extreme performance testing, will be about ~2 us.
2014-08-07 03:52:16 +00:00
2020-06-06 16:40:36 -06:00
The worst case latency is kept stable with no extreme peaks, in spite of any sudden extreme pressure. I have a blog post regarding comparing worst case latency for g3log and other loggers which might be of interest.
2015-07-06 22:32:05 -06:00
You can find it here: https://kjellkod.wordpress.com/2015/06/30/the-worlds-fastest-logger-vs-g3log/
2020-06-06 15:39:27 -07:00
# <a name="feedback">Feedback</a>
2014-08-07 03:52:16 +00:00
If you like this logger (or not) it would be nice with some feedback. That way I can improve g3log and g2log and it is also nice to see if someone is using it.
2020-06-06 15:39:27 -07:00
If you have ANY questions or problems please do not hesitate in contacting me on my blog
2014-08-07 03:56:44 +00:00
http://kjellkod.wordpress.com/2011/11/17/kjellkods-g2log-vs-googles-glog-are-asynchronous-loggers-taking-over
2015-07-06 22:25:45 -06:00
or at ```Hedstrom at KjellKod dot cc` ``
2014-08-07 03:56:44 +00:00
2020-06-06 15:39:27 -07:00
# <a name="say-thanks">Say Thanks</a>
2016-08-18 00:37:28 -06:00
This logger is available for free and all of its source code is public domain. A great way of saying thanks is to send a donation. It would go a long way not only to show your support but also to boost continued development.
[![Donate ](https://img.shields.io/badge/Donate-PayPal-green.svg )](https://www.paypal.me/g3log/25)
* $5 for a cup of coffee
2020-06-06 15:39:27 -07:00
* $10 for pizza
2016-08-18 00:37:28 -06:00
* $25 for a lunch or two
* $100 for a date night with my wife (which buys family credit for evening coding)
* $$$ for upgrading my development environment
* $$$$ :)
2014-08-07 03:52:16 +00:00
2014-08-07 03:57:31 +00:00
Cheers
2015-07-06 22:25:45 -06:00
Kjell *(a.k.a. KjellKod)*