anylogger 1.1.0-beta.2
Get a logger. Any logger.
The logger for libraries
When we want to do logging from a library, we don't want to force the choice of
logging framework on the application developer. Instead, we want to use whatever
logging framework the application developer selected. anylogger
let's you do
just that.
Quickstart
Library | | | App with ulog | |
---|
Install
npm i -D anylogger
Use
import anylogger from 'anylogger'
const log = anylogger('my-library')
log('Anylogger is easy!')
Add to peerDependencies and install
an adapter as dev dependencies.
| Install
npm i -P anylogger
debug anylogger-debug
Add to entry point
index.js
import adapter from "anylogger-debug"
import anylogger from 'anylogger'
adapter(anylogger)
Use
import anylogger from 'anylogger'
const log = anylogger('my-app')
log('Anylogger is easy!')
| Install
npm i -P anylogger
loglevel anylogger-loglevel
Add to entry point
index.js
import adapter from "anylogger-loglevel"
import anylogger from 'anylogger'
adapter(anylogger)
Use
import anylogger from 'anylogger'
const log = anylogger('my-app')
log('Anylogger is easy!')
| Install
npm i -P anylogger
ulog
Add to entry point
index.js
import adapter from "ulog"
import anylogger from 'anylogger'
adapter(anylogger)
Use
import anylogger from 'anylogger'
const log = anylogger('my-app')
log('Anylogger is easy!')
| Install
npm i -P anylogger
log4js anylogger-log4js
Add to entry point
index.js
import adapter from "anylogger-log4js"
import anylogger from 'anylogger'
adapter(anylogger)
Use
import anylogger from 'anylogger'
const log = anylogger('my-app')
log('Anylogger is easy!')
|
What is this?
A logging facade
We, the Javascript community, really need a logging facade. There are dozens
of logging libraries around and we library authors face a challenge. Which logger
do we pick? Should we make this configurable? Should we just not log? Use the
console directly? How do we deal with this complexity?
In software architecture, a
facade hides a complex system
behind a simple interface. In the context of our logging problem, we can have
our library log to a simple facade object. In our application we back the
facade object by the actual logging framework with an
adapter.
So what we need is a simple and small logging facade and a bunch of adapters
for popular loggers.
Introducing anylogger
A tiny ~259 bytes logging facade that you can include in your
library to support logging, while at the same time allowing application
developers to plug in any logging framework they choose.
Instead of building in your own library specific configuration mechanism,
or forcing the choice for a certain logging framework on your users,
or just abandoning logging altogether, choose anylogger
and for just
~259 bytes shared between all libraries doing this, we can
plug in any framework of our choice and all libraries will automatically
start to use that framework. Wouldn't it be much better and easier?
Download
CDN
index.html
<script src="https://unpkg.com/anylogger@1.1.0-beta.2"></script>
<script>(function(){
var log = anylogger('index.html')
log.info('Logging is simple!')
})()</script>
Note that anylogger by default does nothing. You need an adapter to see output.
For example, to send all logging to the console, you can use
anylogger-console
.
Install
Always only install anylogger in application projects. In library projects,
install anylogger as a dev dependency and make it a peer dependency.
Install in an application project
If you are building an application project and have selected a logging
framework, install anylogger, the selected logging framework and the
adapter for that logging framework if needed.
For anylogger-console:
npm install --save anylogger anylogger-console
For debug:
npm install --save anylogger debug anylogger-debug
See anylogger-debug
For loglevel:
npm install --save anylogger loglevel anylogger-loglevel
See anylogger-loglevel
For ulog
npm install --save anylogger ulog
No adapter is needed for ulog
For log4js:
npm install --save anylogger log4js anylogger-log4js
See anylogger-log4js
Check out all
available adapters.
Install in a library project
If you are building a library, install anylogger as a dev-dependency:
npm install --save-dev anylogger
This will add anylogger
as a dev-dependency to your package.json
.
To ensure that application projects using your library also install
anylogger
, make it a peer dependency. Add this to package.json:
"peerDependencies": {
"anylogger": "1.x || >=1.1.0-beta || >=1.2.0-beta || >=1.3.0-beta || >=1.4.0-beta || >=1.5.0-beta || >=1.6.0-beta || >=1.7.0-beta || >=1.8.0-beta || >=1.9.0-beta"
}
The list of (future) betas is an annoyance in the way NPM deals with beta
tags unfortunately. IMHO it is clear that 1.1.0-beta
matches 1.x
and
should thus be covered by it, but alas, NPM thinks otherwise.
If you want your library to also work with possible future betas of
anylogger, then include this list of betas (recommended).
Add anylogger to your install instructions in your project README.md:
Install
npm install --save my-library anylogger
Install dev dependencies in a library project
If you are building a library, you can use the logging framework
you prefer without tightly coupling your library to it by installing
that library and the adapter for it if needed as development dependencies:
For anylogger-console:
npm install --save-dev anylogger-console
For debug:
npm install --save-dev debug anylogger-debug
See anylogger-debug
For loglevel:
npm install --save-dev loglevel anylogger-loglevel
See anylogger-loglevel
For ulog:
npm install --save-dev ulog
No adapter is needed for ulog
For log4js:
npm install --save-dev log4js anylogger-log4js
See anylogger-log4js
Use
Depending on the type of project, either just use anylogger,
or also include the adapter.
Use in a library
In your library code, only use anylogger and restrict yourself to the
Anylogger API to stay framework-independent:
import
my-library.js
import anylogger from 'anylogger'
const log = anylogger('my-library')
log.info('Logging is easy!')
require
my-library.js
const anylogger = require('anylogger')
const log = anylogger('my-library')
log.info('Logging is easy!')
This way, your library does not get tightly coupled to any specific logger
Use in tests for your library
If you have installed an adapter as a dev dependency, you can use that adapter
in your tests so you get the logging the way you like it in your tests.
import
my-library.test.js
import 'anylogger-console'
import anylogger from 'anylogger'
const log = anylogger('my-lbrary:test')
log.info('Logging is easy!')
require
my-library.test.js
require('anylogger-console')
const anylogger = require('anylogger')
const log = anylogger('my-lbrary:test')
log.info('Logging is easy!')
Use in an application project
Install anylogger, your logging framework of choice and an adapter
if needed as regular dependencies. Then, in your main entry point,
include the adapter along with anylogger:
import
main.js
import adapter from 'anylogger-console'
import anylogger from 'anylogger'
adapter(anylogger)
const log = anylogger('my-app')
log.info('Logging is easy!')
require
main.js
const adapter = require('anylogger-console')
const anylogger = require('anylogger')
adapter(anylogger)
const log = anylogger('my-app')
log.info('Logging is easy!')
In your other application modules, use only anylogger and restrict yourself to
the Anylogger API to stay framework-independent:
import
my-module.js
import anylogger from 'anylogger'
const log = anylogger('my-module')
log.info('Logging is easy!')
require
my-module.js
const anylogger = require('anylogger')
const log = anylogger('my-module')
log.info('Logging is easy!')
By limiting yourself to the anylogger API, you ensure that even the code in
your aplication modules remains logging framework independent. That way,
you can easily turn an application module into a library later should the
need arise. Or should you want to switch logging libraries, you will be able
to do that by only changing that one single line that imports the adapter
in the entrypoint in main.js or whatever your main file is called.
Using anylogger for logging
Anylogger is very natural to use:
import anylogger from 'anylogger'
const log = anylogger('my-module')
log('A log message')
log('debug', 'A debug message')
log('warn', 'A warning message')
log.info('Starting...')
log.warn('Watch out!')
log.error('Oh no!! Something went wrong')
Object.keys(anylogger.levels)
if (log.enabledFor('warn')) {
log.warn(expensiveArguments())
}
Anylogger API
So what does this API look like?
anylogger
function anylogger(name: string) => Logger
The main function to call to get a logger.
Accepts the logger name as argument.
name
The name of the logger. String.
The recommended format is <package-name>[:<sub>[:<sub> [..]]]
,
as this is the convention
used by the highly popular debug
module. But you are free to pick
any name you want as long as you steer clear of the characters '*'
,
'='
and ';'
in the logger name.
returns
The existing logger with that name, or a newly created one.
The returned logger adheres to the Logger
type:
This means it is a
LogFunction
and a LogObject
all at once. The LogFunction
aspect makes
anylogger loggers compatible with debug
out of the box, whereas the
LogObject
aspect makes them compatible with the console and most other
logging libraries. I call that a Win-Win!
Logging API
The logger returned by anylogger
is a function that can
do logging on it's own:
log('message')
log('info', 'message')
In addition, the logger looks like a simple console object:
log.debug('message')
log.info('message')
Because of this, the logger created by anylogger is compatible with most
logging frameworks out there, which mostly use one or both of these approaches.
The main API looks like this (in pseudo code):
log: function([level='log'], ...args)
log.error: function(...args)
log.warn: function(...args)
log.info: function(...args)
log.log: function(...args)
log.debug: function(...args)
log.trace: function(...args)
log.enabledFor: function(level) => truthy or falsey
And that's about it. However this covers the basic logging needs.
Note that all logging methods here are part of the upcoming
Console standard, but not all platforms
and frameworks support all of them. In particular the debug
method is not
available everywhere. Anylogger will make sure that the debug
function is
polyfilled if needed.
Is your logging framework not supported? No fear, just...
Write an anylogger adapter
To write an anylogger adapter, you need to make a project that includes both
anylogger and the logging framework the adapter is for as peer dependencies.
You then need to modify one or more of the
anylogger extension points
so the created loggers will be compliant with both the anylogger
Logging API as well as with the logging framework's own API.
It is recommended you call your library anylogger-[adapter]
, where
[adapter]
should be replaced with the name of the logging framework
the adapter is for. For example, the adapter for debug
is called
anylogger-debug
.
In addition, it is recommended you add the keyword "anylogger"
to the
package.json file of your adapter project, so it will show up in the list of
available adapters.
anylogger extension points
The process of logger creation and invocation is split up in such a way
as to optimize possible extension points allowing extensions to re-use
anylogger functionality and avoid having to duplicate code. The extension
points are:
anylogger.levels
anylogger.levels = {error:1, warn:2, info:3, log:4, debug:5, trace:6}
An object containing a mapping of level names to level values.
To be compliant with the anylogger API, loggers should support at least
the log methods corresponding to the default levels, but they may define
additional levels and they may choose to use different numeric values
for all the levels.
The guarantees the Anylogger API makes are:
- there is a logging method corresponding to each level listed in
anylogger.levels
- the levels
error
, warn
, info
, log
, debug
and trace
are always there - each level corresponds to a numeric value
Note that the Anylogger API explicitly does not guarantee that all levels
have distinct values or that the numeric values will follow any pattern
or have any specific order. For this reason it is best to think of levels
as separate log channels, possibly going to different output locations.
You can replace or change this object to include levels corresponding with
those available in the framework you are writing an adapter for. Please
make sure to always include the default levels as well so all code can
rely on the 6 console methods error
, warn
, info
, log
, debug
and
trace
to always be there.
anylogger.new
anylogger.new(name, options) => logger
Creates a new logger function that calls anylogger.log
when invoked.
Uses new Function(..)
to create a named function so that function.name
corresponds to the module name given. Polyfills function.name on platforms
where it is not natively available.
name
The name of the new logger. String. Required.
options
An optional options object. Object. Optional.
If the logging framework you are writing an adapter for uses an options
object, you should override anylogger.new
and do something useful with
the options object here (set it as a property on the logger for example),
because the default implementation just ignores it.
Instead of completely trying to replace the original method, I recommend you
chain it to include your one-time customizations like this:
import anylogger from 'anylogger'
const make = anylogger.new
anylogger.new = (name, options) => {
var logger = make(name, options)
logger.options = options
return logger
}
All anylogger methods are independent of this
so they can all be easily chained
If you need to re-apply customizations any time relevant config changes (such
as active log level changing), override anylogger.ext
.
anylogger.ext
anylogger.ext(logger) => logger
Called when a logger needs to be extended, either because it was newly
created, or because it's configuration or settings changed in some way.
This method must ensure that a log method is available on the logger for
each level in anylogger.levels
.
When overriding anylogger.ext
, please ensure the function can safely
be called multiple times on the same object
logger
The logger that should be (re-)extended. Function. Required.
The default implementation loops over the
anylogger.levels
and creates noop methods for each level.
Additionally it creates a noop enabledFor
that always returns undefined
.
You can override or chain this method to change the way the log methods are
(re-)created. In a library that supports log levels, all methods corresponding
to log levels equal to or higher than the currently active level might be
replaced with console methods instead. Or maybe the destination of the log
messages might change dynamically based on configuration. Apply such changes
in anylogger.ext
as it will be called again whenever relevant config changes.
This allows adapters to (re-)extend the logger so that the new configuration
takes effect.
You may need to ensure in your adapter that anylogger.ext
is called
whenever relevant config changes. By hooking into setters for example.
anylogger.log
anylogger.log([level='log'], ...args)
The log function returned by anylogger calls anylogger.log
, which determines
the log level and invokes the appropriate log method.
Please have a look at the
source
it should make it more clear how to write an adapter. Also consider studying
the available adapters
and learn by example.
Give something back
If you wrote an anylogger
adapter, make sure to share it back with the
community. Publish it to NPM for all to use!
Credits
Credits go to these people, who helped with this project:
Issues
Add an issue in this project's
issue tracker
to let me know of any problems you find, or questions you may have.
Copyright
© 2020 by Stijn de Witt. Some rights reserved.
Contributions by Jakub Jirutka.
License
Licensed under the MIT Open Source license.
gzip-size
The GZIP algorithm is available in different flavours and with different
possible compression settings. The sizes quoted in this README have been
measured using gzip-size
by Sindre Sorhus, your mileage may vary.