A smart logging tool, defined to be minimalist and simple.
To be added
To be added
Displays a info message
Displays a error message
Displays a debug message
Displays a warning message
import qlog from '@contaquanto/qlog';
// Main scope with fields
let log = qlog.scope('MAIN').addFields({
hue: 'br',
a: 1,
stack: 'trace'
});
log.info('INFO MESSAGE');
log.debug('DEBUG MESSAGE');
log.warn('WARNING MESSAGE');
log.error('ERROR MESSAGE');
The slog output is expected to be like this:
2019-09-16T15:35:52-03:00 | I | IO | REQ001 | MAIN > Call0 > Call1 > Call2 | test.go:38 | Doing some IO | {"arg0":"MyArg0","huebr":"call1arg","pop":"abcde"}
There are some fields that are optional (like filename/line number) but it should always follow the same pattern:
DATETIME | LEVEL | OPERATION | TAG | SCOPE | [FILENAME:LINE NUMBER] | MESSAGE | LOG FIELDS
DATETIME
=> An ISO Datetime when the log is displayedLEVEL
=> The level of the log lineI
=> INFO - Shows an information usually to track what's happening inside an applicationW
=> WARN - Shows an warning regarding something that went in a way that might require some attentionE
=> ERROR - Shows an application error that can be expected or notD
=> DEBUG - Shows some debug information to help tracking issuesF
=> FATAL - Shows an error that will quit the application in that point
TAG
=> Line log tag. Use this for tracking related log lines. For example with a HTTP Request IDSCOPE
=> The scope of the current log. Use this to trace the chain of calls inside the application. For example in a context changeFILENAME: LINE NUMBER
=> OPTIONAL When ShowLines is enabled, it will show the filename and the line number of the caller of the slog library. Use this on debug mode to see which piece of code called the log library. Disabled by defaultMESSAGE
=> The messageLOG FIELDS
=> When an instance is created usingaddFields
call, the fields will be serialized to either JSON or Key-Value depending on the configuration of the log instance. Defaults to JSON
The library implements the concept of operation type. This describes which type of operation the log line represents.
IO
=> Anytime you do a I/O Operation such as Database or File Read/WriteAWAIT
=> Anytime you will do a asynchronous operation that will block the flowDONE
=> After anyAWAIT
operation you should always use aDONE
to log that the operation that did theAWAIT
log line has finished.NOTE
=> Also know as Verbose, that operation is used when you're just letting the log reader some note about the operationMSG
=> Common messages such like normal information (which does not fit in other operations) There are some syntax sugars to make easy to use Log Levels with Log Operations together:- Warning Messages
warnDone
=> Same aslog.operation(DONE).warn(message)
warnNote
=> Same aslog.operation(NOTE).warn(message)
warnAwait
=> Same aslog.operation(AWAIT).warn(message)
warnSuccess
=> Same aslog.operation(DONE).warn(message)
warnIO
=> Same aslog.operation(IO).warn(message)
- Error Messages
errorDone
=> Same aslog.operation(DONE).error(message)
errorNote
=> Same aslog.operation(NOTE).error(message)
errorAwait
=> Same aslog.operation(AWAIT).error(message)
errorSuccess
=> Same aslog.operation(DONE).error(message)
errorIO
=> Same aslog.operation(IO).error(message)
- Debug Messages
debugDone
=> Same aslog.operation(DONE).debug(message)
debugNote
=> Same aslog.operation(NOTE).debug(message)
debugAwait
=> Same aslog.operation(AWAIT).debug(message)
debugSuccess
=> Same aslog.operation(DONE).debug(message)
debugIO
=> Same aslog.operation(IO).debug(message)
Use these syntax sugars whenever is possible, instead of callingoperation(X).level
directly.
If a multiline log is displayed, the library will correctly ident all the messages:
2019-09-16T15:39:42-03:00 | I | MSG | REQ001 | MAIN | Multiline call
Thats the second line
Thats the third line | {}
- After calling a
await
, you should always call adone
orsuccess
- Don't add extensive fields to
addFields
as it will pollute the log - Avoid multiline logs as this make parsing hard.
- Avoid using pipes
|
in your log message or fields - Instead of
operation(AWAIT).warn
usewarnAwait
- Use
tag
to indentify calls in the same flow (for example on a HTTP Request) - All
LogInstance
calls returns a newLogInstance
with the modified data. It will never change it's parent data making it completely immutable.