UNPKG

@appium/support

Version:

Support libs used across Appium packages

174 lines (156 loc) 4.48 kB
import _ from 'lodash'; import {supportsColor} from 'supports-color'; import {Console as NodeConsole} from 'console'; import '@colors/colors'; import symbols from 'log-symbols'; import {Writable} from 'stream'; /** * Stream to nowhere. Used when we want to disable any output other than JSON output. */ class NullWritable extends Writable { // eslint-disable-next-line promise/prefer-await-to-callbacks _write(chunk, encoding, callback) { setImmediate(callback); } } /** * A particular console/logging class for Appium's CLI. * * - By default, uses some fancy symbols * - Writes to `STDERR`, generally. * - In "JSON mode", `STDERR` is squelched. Use {@linkcode Console.json} to write the JSON. * * DO NOT extend this to do anything other than what it already does. Download a library or something. */ export class CliConsole { /** * Internal console * @type {globalThis.Console} */ #console; /** * Whether or not to use fancy symbols when logging. * @type {boolean} * */ #useSymbols; /** * Whether or not to use color. */ #useColor; /** * @type {Record<keyof typeof symbols,keyof Extract<import('@colors/colors').Color, 'string'>>} */ static symbolToColor = { success: 'green', info: 'cyan', warning: 'yellow', error: 'red', }; /** * * @param {ConsoleOpts} opts */ constructor({jsonMode = false, useSymbols = true, useColor} = {}) { this.#console = new NodeConsole(process.stdout, jsonMode ? new NullWritable() : process.stderr); this.#useSymbols = Boolean(useSymbols); this.#useColor = Boolean(useColor ?? supportsColor(process.stderr)); } /** * Wraps a message string in breathtaking fanciness * * Returns `undefined` if `msg` is `undefined`. * @param {string} [msg] - Message to decorate, if anything * @param {keyof typeof CliConsole['symbolToColor']} [symbol] - Symbol to use * @returns {string|undefined} */ decorate(msg, symbol) { if (_.isString(msg)) { let newMsg = /** @type {string} */ (msg); if (_.isString(symbol) && this.#useSymbols) { newMsg = `${symbols[symbol]} ${newMsg}`; if (this.#useColor) { newMsg = newMsg[CliConsole.symbolToColor[symbol]]; } } return newMsg; } return msg; } /** * Writes to `STDOUT`. Must be stringifyable. * * You probably don't want to call this more than once before exiting (since that will output invalid JSON). * @param {import('type-fest').JsonValue} value */ json(value) { this.#console.log(JSON.stringify(value)); } /** * General logging function. * @param {string} [message] * @param {...any} args */ log(message, ...args) { this.#console.error(message, ...args); } /** * A "success" message * @param {string} [message] * @param {...any} args */ ok(message, ...args) { this.#console.error(this.decorate(message, 'success'), ...args); } /** * Alias for {@linkcode Console.log} * @param {string} [message] * @param {...any} args */ debug(message, ...args) { this.log(message, ...args); } /** * Wraps {@link console.dir} * @param {any} item * @param {import('util').InspectOptions} [opts] */ dump(item, opts) { this.#console.dir(item, opts); } /** * An "info" message * @param {string} [message] * @param {...any} args */ info(message, ...args) { this.log(this.decorate(message, 'info'), ...args); } /** * A "warning" message * @param {string} [message] * @param {...any} args */ warn(message, ...args) { this.log(this.decorate(message, 'warning'), ...args); } /** * An "error" message * @param {string} [message] * @param {...any} args */ error(message, ...args) { this.log(this.decorate(message, 'error'), ...args); } } /** * Options for {@linkcode CliConsole}. * * @typedef ConsoleOpts * @property {boolean} [jsonMode] - If _truthy_, suppress all output except JSON (use {@linkcode CliConsole#json}), which writes to `STDOUT`. * @property {boolean} [useSymbols] - If _falsy_, do not use fancy symbols. * @property {boolean} [useColor] - If _falsy_, do not use color output. If _truthy_, forces color output. By default, checks terminal/TTY for support via pkg `supports-color`. Ignored if `useSymbols` is `false`. * @see https://npm.im/supports-color */ export const console = new CliConsole(); export {symbols};