Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/logtown/logtown
🦄 Simple Logging Facade for JavaScript
https://github.com/logtown/logtown
javascript logging logging-facade logtown
Last synced: about 2 months ago
JSON representation
🦄 Simple Logging Facade for JavaScript
- Host: GitHub
- URL: https://github.com/logtown/logtown
- Owner: logtown
- License: mit
- Created: 2016-11-04T19:02:32.000Z (about 8 years ago)
- Default Branch: master
- Last Pushed: 2020-06-02T18:32:34.000Z (over 4 years ago)
- Last Synced: 2024-11-10T02:36:44.865Z (about 2 months ago)
- Topics: javascript, logging, logging-facade, logtown
- Language: JavaScript
- Homepage:
- Size: 12.7 MB
- Stars: 32
- Watchers: 4
- Forks: 2
- Open Issues: 1
-
Metadata Files:
- Readme: README.md
- License: LICENSE
Awesome Lists containing this project
- awesome-node-esm - logtown - simple Logging Facade for JavaScript. (Packages / Logging)
README
# Logtown
> Simple Logging Facade for JavaScript.
[![npm version](https://badge.fury.io/js/logtown.svg)](https://www.npmjs.com/package/logtown)
[![Build Status](https://travis-ci.org/logtown/logtown.svg?branch=master)](https://travis-ci.org/logtown/logtown)
[![Known Vulnerabilities](https://snyk.io/test/github/logtown/logtown/badge.svg)](https://snyk.io/test/github/logtown/logtown)
[![Package Quality](http://npm.packagequality.com/shield/logtown.svg)](http://packagequality.com/#?package=logtown)
[![Code Climate](https://codeclimate.com/github/logtown/logtown/badges/gpa.svg)](https://codeclimate.com/github/logtown/logtown)
[![Greenkeeper badge](https://badges.greenkeeper.io/logtown/logtown.svg)](https://greenkeeper.io/)[![dependencies:?](https://img.shields.io/david/logtown/logtown.svg)](https://david-dm.org/logtown/logtown)
[![devDependencies:?](https://img.shields.io/david/dev/logtown/logtown.svg?style=flat)](https://david-dm.org/logtown/logtown)## Installation
```bash
npm install logtown --save
```or
```bash
yarn add logtown
```## Usage
First of all you should notice, that `logtown` is not a *logger*. You can use any logger you want underneath, there are a
lot of great tools like intel, winston and others.
And if you don't define any wrappers to use you won't see any output. There are 2 ready for use wrappers included into the
package as an example, but you can use your own in any time.So let's start from simple use case:
```javascript
const Logger = require('logtown');// Adding wrapper is very simple, you need implement at least 1 method from this array ['log', 'silly', 'debug', 'info', 'warn', 'error']
// log method has only 1 difference from others it receives additional *level* parameter
Logger.addWrapper({
log(id, level, stats, ...rest) {
console[level](`[${id}] ${rest.join(' ')}`)
}
})// Once you added wrapper you can use logger in any place of your app
const logger = Logger.getLogger('mymodule-label');
// if you prefer factory method then you can use getLogger as:
const logger = require('logtown')('my-namespace');
// or
// const logger = Logger('my-namespace');logger.silly('Silly message');
logger.debug('Debug message');
logger.info('Info message');
logger.warn('Warn message');
logger.error('Error message');
```### Using in es5 environment
There are several precompiled versions of this module placed in folders `es5` and `es6`. You might select the one you
need by importing `require('logtown/es5/common')` for old commonjs environments.
### Adding new wrapperAdding wrapper as you noticed before in the example, is pretty simple operation. As it was noticed, you need to implement at least one
method or pass single `function` that will work like the most advanced wrapper's function `log`.Example,
```javascript
const Logger = require('logtown');// passing function as wrapper
Logger.addWrapper(function (id, level, stats, ...rest) {
console.log(`${level} [${id}]`, ...rest);
});// the same as above
Logger.addWrapper({
log: function (id, level, stats, ...rest) {
console.log(`${level} [${id}]`, ...rest);
}
});// pass only one function for required level
Logger.addWrapper({
info: function (id, stats, ...rest) {
// log only info messages in this wrapper
console.info(...rest);
}
});class AdvancedWrapper {
constructor(options = {}) {
this.supperLogger = new SupperLogger(options);
// preparing supper logger ...
}
static logOptions = { // In rare cases you may need to avoid passing arguments that were modified in plugin, to tell logtown about that, you can define `logOptions` static object
passInitialArguments: true,
};
log(id, level, stats, ...rest) {
this.supperLogger.log(...rest);
}
}
Logger.addWrapper(new AdvancedWrapper({option1: "value"}));
```## Configuration
Logtown can be configured in 3 ways:
* by defining **application level** configurations
* by defining **logger level** configurations
* and by defining **global level** configurations1. **Application level** can be used by passing configuration object into `Logger.configure({})` static method.
```javascript
const logger = Logger.configure({
namespaces: {'mymodule': {disable: ['debug']}}, // disable one or more level for specific namespace
useGlobal: false, // disable usage of global configurations, it is enabled by default
disable: ['info', 'warn'] // disable globaly specific levels
});
```
2. You can set several options also during creation of the new logger.```javascript
const logger1 = Logger.getLogger('mymodule', {disable: 'debug'})
const logger2 = Logger.getLogger('mymodule', {disable: ['debug', 'info']})
```
Each time you `getLogger` with new configuration, it will be merged with previously defined ones,
even though logger instance will be the same.3. In very specific case you need to shut down loggers from nested npm modules that you can't affect on. One of the solutions
is to use [peerDependencies](https://docs.npmjs.com/files/package.json#peerdependencies), but if peerDependencies is
not the case you can set global configuration object, that will be handled by nested logtown package.
```javascript
_.set(global, 'logtown.namespaces.module-that-makes-mess-in-logs.disable', ['silly', 'debug']);
```
Tip: it is easy to use `easy-loader` to configure logtown for each environment.```javascript
const config = require('../config');
const Logger = require('logtown');Logger.configure(config('logtown', {}))
```## Experimental features
## Custom methods
By default you are limited with defined method set. It gives possibility to rely on logger api anytime and be ready to
swap out it when needed. But sometimes you would need to use your logger's very specific functionality. For example,
`console.group`. There are 2 ways of using it - by direct calling console.group method, that is bad and in some level
unstable. And second one, to notify somehow your wrapper.So `logtown` introduces `send` method. That is behaves the same as other methods, but gives possibility to pass `level`.
```javascript
Logger.addWrapper({
group: function (id, stats, ...rest) {
console.group(id);
},
groupend: function(id, stats, ...rest) {
console.groupEnd(id);
}
});const logger = Logger.getLogger('my-namespace');
logger.send('group');
logger.debug('Hello World');
logger.send('groupend');
```Important to note that `send` method won't fallback to `log` method if it exists in wrapper.
## Introducing plugin API
For instance, you want to add debugging info during testing. Let's say you want to log filename:linenumber with your
favourite winston logger. But you can't([#200](https://github.com/winstonjs/winston/issues/200)). So one of the solutions
would be to switch to [tracer](https://github.com/baryon/tracer), or just add right plugin for your logtown configuration.```javascript
const Logger = require('logtown');const pluginFactoryFn = require('logtown/es6/common/plugins/stacktrace');
Logger.addPlugin(pluginFactoryFn({ prependRest: true }));
```And that's it, now debug stats are available in all your wrappers.
### Creating new plugin
Plugin is a simple function, that receives mutable `context` object. You can modify only 2 fields in it: `context.stats` and
`context.args`. These are the values that will be passing to wrappers. Also there are some additional getters: `context.id`, `context.level` and
`context.arguments`. The last is an array with initial arguments passed. It is can be useful when one your plugins has changed
type or value of argument but you are required to know about it's initial value or type.```js
function plugin(ctx) {
if (ctx.level === 'ERROR' && ctx.args[0] instanceof Error) {
ctx.args[0] = prettyPrint(ctx.args[0]);
}
}
logtown.addPlugin(plugin);
```## Tags
```javascript
const logger = Logger.getLogger('my-namespace', {tags: ['debug', 'lib-1', 'lib-2']});Logger.configure({tags: { disable: ['lib-1'] }});
```## Wrapper options
Since this feature has not been validated in production, it will stay in experimental until it will have enough testing.
## More articles
- https://medium.com/@dzelenetskiy/confident-logging-in-nodejs-and-browser-cb97d91e673d
## License
Logtown is released under the MIT license.
## Donate
[![](https://img.shields.io/badge/patreon-donate-yellow.svg)](https://www.patreon.com/red_rabbit)
[![](https://img.shields.io/badge/flattr-donate-yellow.svg)](https://flattr.com/profile/red_rabbit)