Important: This documentation covers Yarn 1 (Classic).
For Yarn 2+ docs and migration guide, see yarnpkg.com.

Package detail

react-native-logs

mowispace109.1kMIT5.3.0TypeScript support: included

Performance-aware simple logger for React-Native with namespaces, custom levels and custom transports (colored console, file writing, etc.)

log, logs, logger, react-native, expo, error, console, colors, debug, file, custom, levels, namespace

readme

npm GitHub contributions welcome

react-native-logs

Performance-aware simple logger for React-Native, Expo (managed and bare) and react-native-web with custom levels and transports (colored console, file writing, etc.).

Each level has its severity: a number that represents its importance in ascending order from the least important to the most important. Eg. debug:0, info:1, warn:2, error:3.

By config the logger with a minium severity level, you will see only the logs that have it highest. Then logs will be managed by transport: the function that will display/save/send log messages.

It is also possible to extend the logger to create namespaced logs. In this way you will be able to see the log messages only for one or some parts of the code of your choice

Demo console transport with custom colored levels and namespaces: console log demo

Why another logging library?

After trying the most known logging libraries, like winston and bunyan, we found that for react-native we needed something simpler, but still flexible, and without dependencies on nodejs. Comments and suggestions are welcome.

Installation

npm install --save react-native-logs

OR

yarn add react-native-logs

OR

expo install react-native-logs

Quick Start

import { logger } from "react-native-logs";

var log = logger.createLogger();

log.debug("This is a Debug log");
log.info("This is an Info log");
log.warn("This is a Warning log");
log.error("This is an Error log");

By default the createLogger() method (called without arguments) will create a simple console logger with debug, info, warn and error levels.

Configuration

You can customize the logger by passing a config object to the createLogger method (see example below). All params are optional and will take default values if no corresponding argument is passed.

Parameter Type Description Default
severity string Init logs severity (least important level you want to see) debug (or the first custom level)
transport function or [function] The transport function/s for logs (see below for presets) The preset transport consoleTransport
transportOptions Object Set custom options for transports null
levels Object Set custom log levels: {name:power} false
async boolean Set to true for async logs (to improve app performance) true
asyncFunc function Set a cutom async function (cb: Function) => {return cb()} setTimeout
stringifyFunc function Set a cutom stringify function (msg: any) => string a customized JSON.stringify
formatFunc function Set a custom format function (level: string, extension?: string, msg: any) => string default string format function
dateFormat string or function time, local, utc, iso or (date: Date) => string time
printLevel boolean Choose whether to print the log level true
printDate boolean Choose whether to print the log date/time true
fixedExtLvlLength boolean Ensure consistent character count alignment when printing extensions and levels false
enabled boolean Enable or disable logging true
enabledExtensions string[] Enable only certain namespaces null

Example with common configuration

import { logger, consoleTransport } from "react-native-logs";

var log = logger.createLogger({
  levels: {
    debug: 0,
    info: 1,
    warn: 2,
    error: 3,
  },
  severity: "debug",
  transport: consoleTransport,
  transportOptions: {
    colors: {
      info: "blueBright",
      warn: "yellowBright",
      error: "redBright",
    },
  },
  async: true,
  dateFormat: "time",
  printLevel: true,
  printDate: true,
  fixedExtLvlLength: false,
  enabled: true,
});

log.debug("Debug message");
log.info({ message: "hi!" });

Custom levels

Log levels have this format: { name : severity } and you can create your personalized list, Eg:

import { logger } from "react-native-logs";

var log = logger.createLogger({
  levels: {
    trace: 0,
    info: 1,
    silly: 2,
    error: 3,
    mad: 4,
  },
});

log.silly("Silly message");

Levels typing

(available only if you use typescript)

The package will take the types of log levels directly from the configuration, or the default ones if not specified.

import { logger } from "react-native-logs";

var log = logger.createLogger({
  levels: {
    trace: 0,
    info: 1,
    error: 2,
  },
});

log.trace("message"); // correct log call

log.silly("message"); // typescript error, "silly" method does not exist

Custom transport

You can write your own transport to send logs to a cloud service, save it in to a database, or do whatever you want. The following parameters are received by the function:

  • msg: any: the message formatted by logger "[time] | [namespace] | [level] | [msg]"
  • rawMsg: any: the message (or array of messages) in its original form
  • level: { severity: number; text: string }: the log level
  • extension?: string | null: its namespace if it is an extended log
  • options?: any: the transportOptions object

You can define your custom transport as follow (example in typescript):

import { logger, transportFunctionType } from "react-native-logs";

const customTransport: transportFunctionType<{ myCustomOption: string }> = (
  props
) => {
  // Do here whatever you want with the log message
  // You can use any options setted in config.transportOptions
  // Eg. a console log: console.log(props.level.text, props.msg)
};

var log = logger.createLogger({
  transport: customTransport,
  transportOptions: {
    myCustomOption: "option",
  },
});

log.debug("Debug message");

Transport Options

By setting the transportOptions parameter you can insert the options that will be passed to transports. For some transports these may be mandatory, as in the case of the FS option for the fileAsyncTransport (see preset transports list for details).

import { logger, fileAsyncTransport } from "react-native-logs";
import RNFS from "react-native-fs";

var log = logger.createLogger({
  transport: fileAsyncTransport,
  transportOptions: {
    FS: RNFS,
    fileName: `log.txt`,
  },
});

log.debug("Debug message");

Multiple Arguments

Log messages can be concatenated by adding arguments to the log function:

var errorObject = {
  staus: 404,
  message: "Undefined Error",
};
log.error("New error occured", errorObject);

Preset transports

react-native-logs includes some preset transports. You can import the one of your choice: import { logger, <transportName> } from 'react-native-logs';

Example

import { logger, mapConsoleTransport } from "react-native-logs";

var log = logger.createLogger({
  transport: mapConsoleTransport,
});

log.debug("Debug message");

List of included preset transports

consoleTransport

Print the logs with a formatted console.log output.

If you need a different console or method to be used instead of console.log you can set the consoleFunc option with your custom console.

name type description default
colors object If setted you can choose the log colors, defined by level: {level:color} null
extensionColors object If setted you can choose the extension label colors: {extension:color} null
consoleFunc (msg:any)=>any If setted you can choose the console object null

Available colors

name ansi code note
default null default console color
black 30
red 31
green 32
yellow 33
blue 34
magenta 35
cyan 36
white 37
grey 90
redBright 91
greenBright 92
yellowBright 93
blueBright 94
magentaBright 95
cyanBright 96
whiteBright 97

Example

import { logger, consoleTransport } from "react-native-logs";

var log = logger.createLogger({
  levels: {
    debug: 0,
    info: 1,
    warn: 2,
    error: 3,
  },
  transport: consoleTransport,
  transportOptions: {
    colors: {
      info: "blueBright",
      warn: "yellowBright",
      error: "redBright",
    },
    extensionColors: {
      root: "magenta",
      home: "green",
    },
  },
});

var rootLog = log.extend("root");
var homeLog = log.extend("home");

rootLog.info("Magenta Extension and bright blue message");
homeLog.error("Green Extension and bright red message");

mapConsoleTransport

Print the logs with a selected console method (console.log, console.warn, console.error, etc.).

name type description default
mapLevels object Select the console method by level: {level:method} null

If mapLevels is not setted, the transport will try to map the console methods with the level name.

Example

import { logger, mapConsoleTransport } from "react-native-logs";

var log = logger.createLogger({
  levels: {
    debug: 0,
    info: 1,
    warn: 2,
    err: 3,
  },
  transport: mapConsoleTransport,
  transportOptions: {
    mapLevels: {
      debug: "log",
      info: "info",
      warn: "warn",
      err: "error",
    },
  },
});

log.debug("Print this with console.log");
log.err("Print this with console.error");

fileAsyncTransport

This transport requires the installation of react-native-fs(install tutorial here) or expo-file-system, and allows you to save the logs on the <filePath>/<fileName>.txt file.

If you want a new file to be created every day you can use {date-today} in the fileName: app_logs_{date-today}.log -> app_logs_D-M-YYYY.log.

Accepted Options:

name type description default
FS Object MANDATORY, filesystem instance for the transport (RNFS or expo FileSystem) null
fileName string set logs file name (insert {date-today} for current date)
fileNameDateType eu,us,iso {date-today} date type eu "DD-MM-YYYY", us "MM-DD-YYYY", iso "YYYY-MM-DD" eu
filePath string set logs file path RNFS.DocumentDirectoryPath or expo FileSystem.documentDirectory

Example:

import { logger, fileAsyncTransport } from "react-native-logs";
import RNFS from "react-native-fs";
/* EXPO:
 * import * as FileSystem from 'expo-file-system';
 */

let today = new Date();
let date = today.getDate();
let month = today.getMonth() + 1;
let year = today.getFullYear();

var log = logger.createLogger({
  severity: "debug",
  transport: fileAsyncTransport,
  transportOptions: {
    FS: RNFS,
    /* EXPO:
     * FS: FileSystem,
     */
    fileName: `logs_{date-today}`, // Create a new file every day
  },
});

log.info("Print this string to a file");

NOTE: Following this example it will be possible to upload the file to your remote server

sentryTransport

Send logs to Sentry. The transport allows setting which log levels are errors, so that all others are treated as breadcrumbs, meaning log messages related to the next error that will occur. Otherwise, if not set, all messages will be treated as errors.

Accepted Options:

name type description default
SENTRY Object MANDATORY, Sentry instance for the transport null
errorLevels Array<string> or string Specify which log levels are errors (If null, all msg will be treated as errors) null

Example:

import { logger, sentryTransport } from "react-native-logs";
import * as Sentry from "@sentry/react-native";

/*
 * Configure sentry
 */

var log = logger.createLogger({
  severity: "debug",
  transport: sentryTransport,
  transportOptions: {
    SENTRY: Sentry,
    errorLevels: "error",
  },
});

log.warn("Send this log to Sentry as breadcumb");
log.error("Send this log to Sentry as error");

Extensions (Namespaced loggers)

To enable logging only for certain parts of the app, you can extend the logger to different namespaces using the "extend" method. You can enable these extensions from the configuration (config.enabledExtensions) or by using the enable/disable methods.

Example:

import { logger, consoleTransport } from "react-native-logs";

var log = logger.createLogger({
  transport: consoleTransport,
  enabledExtensions: ["ROOT", "HOME"],
});

var rootLog = log.extend("ROOT");
var homeLog = log.extend("HOME");
var profileLog = log.extend("PROFILE");

log.debug("print this"); // this will print "<time> | DEBUG | print this"
rootLog.debug("print this"); // this will print "<time> | ROOT | DEBUG | print this"
homeLog.debug("print this"); // this will print "<time> | HOME | DEBUG | print this"
profileLog.debug("not print this"); // this extension is not enabled

Methods

enable/disable

Dynamically enable/disable loggers and extensions, if it is called without parameters then it will disable/enable the whole logger:

import { logger, consoleTransport } from "react-native-logs";

var log = logger.createLogger({
  transport: consoleTransport,
  enabledExtensions: ["ROOT", "HOME"],
});

var rootLog = log.extend("ROOT");
var homeLog = log.extend("HOME");

log.info("print this"); // this will print "<time> | ROOT | INFO | print this"
homeLog.info("print this"); // extension is enabled

log.disable("HOME");

homeLog.info("not print this"); // extension is not enabled
rootLog.info("print this"); // extension is enabled

log.disable();

homeLog.info("not print this"); // logger is not enabled
rootLog.info("not print this"); // logger is not enabled
log.info("not print this"); // logger is not enabled

getExtensions

Get an array of currently created extensions.

setSeverity

You can set the severity level by passing the name(string) of the least important level you want to see. This method will overwrite any config.severity option set in logger creation.

var log = logger.createLogger();

log.setSeverity("info");
log.debug("This log will not be printed");
log.info("This log will be printed correctly");
log.error("This log will be printed correctly");

getSeverity

You can get the current severity level setted.

var log = logger.createLogger();

var defaultseverity = log.getSeverity(); // severity = debug
log.setSeverity("info");
var severity = log.getSeverity(); // severity = info
log.setSeverity("error");
var newseverity = log.getSeverity(); // newseverity = error

patchConsole

(Experimental)

With this method you are going to overwrite the default console, which is useful in case you are installing this package in an existing software where you use the default console.

Any levels you specify in configuration, if they exist, will be mapped to the console methods (console.log, console.info, console.error, etc...).

If you do not specify a log level in configuration then your first level will be mapped to it.

All calls to console.* will then be handled by your react-native-logs logger and then you can manage their visibility via severity, or change their transports.

This method may have undesirable effects, so I recommend using it only if necessary.

import { logger, consoleTransport } from "react-native-logs";

const config = {
  levels: {
    debug: 0,
    log: 1,
    warn: 2,
    error: 3,
  },
  severity: "log",
  printLevel: true,
  printDate: true,
};

var log = logger.createLogger(defaultConfig);

// this call will use default console
console.log("This method use console");

log.patchConsole();

// all of the following console calls will use your react-native-logs logger
console.log("This method use your logger");
console.warn("This method use your logger too");
console.debug("this message will not be shown"); // severity is set to 'log'

Usage Tips

Logs only in development mode

In reacly-native, after you have create your logger, you can set to log only in development using the __DEV__ as follows:

import {
  logger,
  consoleTransport,
  fileAsyncTransport,
} from "react-native-logs";
import RNFS from "react-native-fs";

const config = {
  transport: __DEV__ ? consoleTransport : fileAsyncTransport,
  severity: __DEV__ ? "debug" : "error",
  transportOptions: {
    colors
    FS: RNFS,
  },
};

var log = logger.createLogger();

This will block all the logs in production, but not the errors, so the app performance will not be affected. This will also change the transport: print to console in development and save to file in production.

Global logger in react-native

In order to have a global logger throughout the app, i recommend using a config.js file to initialize the logger so it can be imported wherever it is needed. Example:

//config.js
import {
  logger,
  consoleTransport,
  fileAsyncTransport,
} from "react-native-logs";
import RNFS from "react-native-fs";

var LOG = logger.createLogger({
  transport: __DEV__ ? consoleTransport : fileAsyncTransport,
  severity: __DEV__ ? "debug" : "error",
  transportOptions: {
    colors: {
      info: "blueBright",
      warn: "yellowBright",
      error: "redBright",
    },
    FS: RNFS,
  },
});

export { LOG };
//index.js and other app files
import { LOG } from "./config";

LOG.info("app log test");

To use extended loggers in all files you can also re-declare them:

//root.js
import { LOG } from "./config";
var log = LOG.extend("ROOT");

log.info("root log test");
//root2.js
import { LOG } from "./config";
var log = LOG.extend("ROOT");

log.info("root log test");
//home.js
import { LOG } from "./config";
var log = LOG.extend("HOME");

log.info("home log test");

Use multiple transports

To use multiple transports by passing it as an Array:

import {
  logger,
  consoleTransport,
  fileAsyncTransport,
  sentryTransport,
  crashlyticsTransport,
  transportFunctionType,
} from "react-native-logs";
import RNFS from "react-native-fs";
import * as Sentry from "@sentry/react-native";
import crashlytics from "@react-native-firebase/crashlytics";

const crashlyticsModule = crashlytics();

var customTransport: transportFunctionType = (props) => {
  // Do here whatever you want with the log message
  // Eg. a console log: console.log(props.level.text, props.msg)
};

const log = logger.createLogger({
  transport: [
    consoleTransport,
    fileAsyncTransport,
    sentryTransport,
    crashlyticsTransport,
    customTransport,
  ],
  transportOptions: {
    FS: RNFS,
    SENTRY: Sentry,
    CRASHLYTICS: crashlyticsModule,
    colors: {
      info: "blueBright",
      warn: "yellowBright",
      error: "redBright",
    },
  },
});

Improve performance

In react-native you can improve performance by setting the InteractionManager.runAfterInteractions async function:

import { logger } from "react-native-logs";

const InteractionManager = require("react-native").InteractionManager;

const log = logger.createLogger({
  async: true,
  asyncFunc: InteractionManager.runAfterInteractions,
});

Sponsors

changelog

[5.3.0] - 25-10-2024

  • Improve type definitions (pr #109 by @DanielSRS)
  • Minor bugfix

BREAKING CHANGES (only for typescript config)

Starting from version v 5.3.0, the definition of types has been improved: transportOptions are now strongly typed based on the specific transport specified in the configuration, and it is no longer necessary to specify log level types, as these are also taken directly from the configuration.

The configuration must be passed inline for it to work correctly and the log level type definitions that needed to be set up until version 5.2.2 must now be removed:

import { logger } from "react-native-logs";

var log = logger.createLogger({
  levels: {
    trace: 0,
    info: 1,
    error: 2,
  },
});

log.trace("message"); // correct log call
log.silly("message"); // typescript error, "silly" method does not exist

Additionally, it is now possible to specify custom options in your custom transport:

const customTransport: transportFunctionType<{ myCustomOption: string }> = (
  props
) => {
  // ...
};

[5.2.2] - 21-10-2024

  • Reverting to the old merge config function

[5.2.1] - 18-10-2024

  • Minor bugfix

[5.2.0] - 17-10-2024

  • Ensures JSON.stringify print nested objects correctly (issue #97)
  • Only merge non undefined config values (pr #105 by @SYoder1)
  • Correct README for Sentry logging (pr #104 by @ssorallen)
  • Add crashlytics transport (pr #91 by @chad-aijinet)
  • Added fileNameDateType option to the file transport for selecting the date format
  • Minor bugfix

[5.1.0] - 26-01-2024

  • Ensures JSON.stringify correctly (Thanks @iago-f-s-e)
  • Added formatFunc option (Thanks @chmac)
  • Added ability to set errorLevels on sentry transport
  • Correct format function type name in default stringify func
  • Added the confg option fixedExtLvlLength, allowing for uniform extension and level lengths by adding spaces, ensuring aligned logs
  • Minor bugfix

[5.0.1] - 04-07-2022

  • Fixed fileName in fileAsyncTranport
  • in fileName now you can pass {date-today}

[5.0.0] - 30-06-2022

  • Simplified init configuration (thanks to @Harjot1Singh)
  • Added levels typing
  • Customizable stringify function
  • Transport config option now accept array of transports
  • fileAsyncTransport can be configured to create a new file everyday
  • customizable console.log function in consoleTrasport
  • Added patchConsole method
  • dateFormat now accept a custom function

BREAKING CHANGES

There are no real breaking changes in this version, only the default async function has been changed, which is now a simple setTimeout to 0 ms.

[4.0.1] - 15-01-2022

  • enable() and disable() methods can now enable or disable extensions

[4.0.0] - 03-01-2022

In this new major update many of the features requested in the previous issues have been fixed, introduced or improved:

  • reversed the extension mechanism, now if they are not specified, they will all be displayed
  • added the ability to choose the colors of the levels for the consoleTransport
  • added the ability to choose the colors of extensions in consoleTransport
  • added a transport that prints logs with the native console methods (log, info, error, etc ...)
  • fixed type exports
  • minor bugfix

BREAKING CHANGES

  • from this version if no extensions are specified in the configuration then all are printed, otherwise only the specified ones
  • the colors option for the consoleTransport must now be set with the desired colors for each level (see the readme), if not set the logs will not be colored
  • removed css web color support (latest chrome versions support ansi codes)

[3.0.4] - 04-06-2021

  • queue management to avoid race conditions problems with ExpoFS
  • minor bugfix

[3.0.3] - 12-02-2021

  • removed EncodingType reference on fileAsyncTransport

[3.0.2] - 27-01-2021

  • fixed web colors in console transport

[3.0.1] - 27-01-2021

  • fixed ansi colors in console transport

[3.0.0] - 26-01-2021

This new version introduces many changes, the log management has been modified to allow the creation of namespaced loggers and to simplify the creation of custom transports. The creation of namespaced loggers is done via the "extend" function on the main logger. This makes it possible to enable or disable logging only for certain parts of the app. The extend function is for now only enabled at the first level, it is not possible to extend an already extended logger in order to avoid loops in the controls that would affect performance.

  • complete refactoring
  • added namespaced logs via extend function!
  • expofs support for file transport (beta)
  • sentry transport
  • logs concatenation on single line
  • bugfix

BREAKING CHANGES

To upgrade to version 3 you need to change the logger creation. The default transports have now been reduced, but they support the same functions as before but through options, e.g. to get asynchronous logs you can set the async:true option instead of importing a special transport. Custom transports also need to change, they now receive a single "props" parameter containing everything you need, the message formatting has been moved out of the transport so you can just output it. It is still possible to format the logs at will. Please refer to the new documentation for details.

[2.2.1] - 23-05-2020

  • added "ansiColorConsoleSync" transport to color logs on terminal (and VScode terminal)

[2.2.0] - 11-05-2020

  • added log messages concatenation "log(msg1,msg2,etc...)"
  • added dataFormat transportOptions (thanks @baldur)
  • bugfix

[2.1.2] - 14-04-2020

  • fixed bug RNFS wrong require line (thanks @jbreuer95)

[2.1.0] - 08-04-2020

  • added possibility to pass options to transport with transportOptions property

[2.0.2] - 13-03-2020

  • bugfix

[2.0.1] - 04-03-2020

  • remove transport export from main index module to avoid require errors

[2.0.0] - 23-02-2020

  • added preset file transport based on react-native-fs
  • added preset transport with react-native AfterInteractions
  • bugfix

Breaking Changes

  • removed parameter cb() from transport functions
  • preset transport renamed

[1.0.2] - 12-07-2020

  • bugfix

[1.0.1] - 12-07-2020

  • npm release

[1.0.0] - 12-07-2020

  • initial commit