xen-orchestra/@xen-orchestra/log
2024-02-13 17:35:58 +01:00
..
transports chore(log): prefix internal modules by _ 2023-01-15 12:59:31 +01:00
_compileGlobPattern.js feat(log): respect env.{DEBUG,NODE_DEBUG} by default 2022-11-18 10:42:13 +01:00
_compileGlobPattern.test.js chore(log/utils.test.js): rename to _compileLogPattern.test.js 2023-01-15 12:59:31 +01:00
_createTransport.js chore(log): move createTransport into own module 2023-01-15 12:59:31 +01:00
_levels.js chore(log): prefix internal modules by _ 2023-01-15 12:59:31 +01:00
_levels.test.js chore(log): prefix internal modules by _ 2023-01-15 12:59:31 +01:00
_Log.js chore(log): move Log into own module 2023-01-15 12:59:31 +01:00
_utils.js chore(log): prefix internal modules by _ 2023-01-15 12:59:31 +01:00
.npmignore feat: unified .npmignore for all packages 2021-04-07 13:58:14 +02:00
.USAGE.md docs(log): document method signature 2024-02-13 17:35:58 +01:00
configure.js chore(log): move createTransport into own module 2023-01-15 12:59:31 +01:00
dedupe.js feat(log/dedupe): helper to remove duplicated logs 2023-01-15 12:59:31 +01:00
index.js chore(log): move Log into own module 2023-01-15 12:59:31 +01:00
package.json feat: technical release (#6645) 2023-01-27 16:16:26 +01:00
README.md docs(log): document method signature 2024-02-13 17:35:58 +01:00

@xen-orchestra/log

Package Version License PackagePhobia Node compatibility

Logging system with decoupled producers/consumer

Install

Installation of the npm package:

npm install --save @xen-orchestra/log

Usage

Producers

Everywhere something should be logged:

import { createLogger } from '@xen-orchestra/log'

const log = createLogger('my-module')

log.debug('only useful for debugging')
log.info('this information is relevant to the user')
log.warn('something went wrong but did not prevent current action')
log.error('something went wrong')
log.fatal('service/app is going down')

// you can add contextual info
log.debug('new API request', {
  method: 'foo',
  params: [ 'bar', 'baz' ]
  user: 'qux'
})

// by convention, errors go into the `error` field
log.error('could not join server', {
  error,
  server: 'example.org',
})

A logging method has the following signature:

interface LoggingMethod {
  (error): void

  (message: string, data?: { error?: Error; [property: string]: any }): void
}

Consumer

Then, at application level, configure the logs are handled:

import { createLogger } from '@xen-orchestra/log'
import { configure, catchGlobalErrors } from '@xen-orchestra/log/configure'
import transportConsole from '@xen-orchestra/log/transports/console'
import transportEmail from '@xen-orchestra/log/transports/email'

const transport = transportEmail({
  service: 'gmail',
  auth: {
    user: 'jane.smith@gmail.com',
    pass: 'H&NbECcpXF|pyXe#%ZEb',
  },
  from: 'jane.smith@gmail.com',
  to: ['jane.smith@gmail.com', 'sam.doe@yahoo.com'],
})

configure([
  {
    filter: process.env.DEBUG,

    transport: transportConsole(),
  },
  {
    // only levels >= warn
    level: 'warn',

    transport,
  },
  {
    type: 'email',

    service: 'gmail',
    auth: {
      user: 'jane.smith@gmail.com',
      pass: 'H&NbECcpXF|pyXe#%ZEb',
    },
    from: 'jane.smith@gmail.com',
    to: ['jane.smith@gmail.com', 'sam.doe@yahoo.com'],
  },
])

// send all global errors (uncaught exceptions, warnings, unhandled rejections)
// to this logger
catchGlobalErrors(createLogger('app'))

A transport as expected by configure(transport) can be:

  • a function that will receive emitted logs;
  • an object with a type property and options which will be used to create a transport (see next section);
  • an object with a nested transport which will be used if one of the following conditions is fulfilled:
    • filter: pattern which is matched against the log namespace (can also be an array of filters);
    • level: the minimal level of accepted logs;
  • an array of transports.

Transports

Console

import transportConsole from '@xen-orchestra/log/transports/console'

configure(transportConsole())

Email

Optional dependency:

> yarn add nodemailer pretty-format

Configuration:

import transportEmail from '@xen-orchestra/log/transports/email'

configure(
  transportEmail({
    service: 'gmail',
    auth: {
      user: 'jane.smith@gmail.com',
      pass: 'H&NbECcpXF|pyXe#%ZEb',
    },
    from: 'jane.smith@gmail.com',
    to: ['jane.smith@gmail.com', 'sam.doe@yahoo.com'],
  })
)

Syslog

Optional dependency:

> yarn add split-host syslog-client

Configuration:

import transportSyslog from '@xen-orchestra/log/transports/syslog'

// By default, log to udp://localhost:514
configure(transportSyslog())

// But TCP, a different host, or a different port can be used
configure(transportSyslog({ target: 'tcp://syslog.company.lan' }))

Helpers

Dedupe

Wraps a transport to limit the number of duplicate logs.

import { dedupe } from '@xen-orchestra/log/dedupe'

configure(
  dedupe({
    timeout: 500e3, // default to 600e3, ie 10 minutes
    transport: console.log,
  })
)

Duplicate logs will be buffered for timeout milliseconds or until a different log is emitted, at which time a dedicated log entry is emitted which indicate the number of duplicates that occured.

const logger = createLogger('app')

// Log some duplicate messages
logger.error('Something went wrong')
logger.error('Something went wrong')
logger.error('Something went wrong')

// Log a different message
logger.info('This is a different message')

In this example, the first three log entries are identical and the last two will be treated as duplicates. They will be grouped together and sent to the transport as a single log with nDuplicates: 2 data when a different log entry is emitted (or after the timeout as elasped if there weren't one).

The output in the console would look something like:

app ERROR Something went wrong
app ERROR duplicates of the previous log were hidden { nDuplicates: 2 }
app INFO This is a different message

Contributions

Contributions are very welcomed, either on the documentation or on the code.

You may:

  • report any issue you've encountered;
  • fork and create a pull request.

License

ISC © Vates SAS