2021-03-25 07:00:22 +00:00
|
|
|
|
import type { CompileError } from './parser/utils/error.js';
|
2021-03-17 19:48:49 +00:00
|
|
|
|
import { bold, blue, red, grey, underline } from 'kleur/colors';
|
|
|
|
|
import { Writable } from 'stream';
|
2021-03-19 21:07:45 +00:00
|
|
|
|
import { format as utilFormat } from 'util';
|
2021-03-17 19:48:49 +00:00
|
|
|
|
|
|
|
|
|
type ConsoleStream = Writable & {
|
2021-03-19 21:07:45 +00:00
|
|
|
|
fd: 1 | 2;
|
2021-03-17 19:48:49 +00:00
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
export const defaultLogDestination = new Writable({
|
|
|
|
|
objectMode: true,
|
|
|
|
|
write(event: LogMessage, _, callback) {
|
|
|
|
|
let dest: ConsoleStream = process.stderr;
|
2021-03-19 21:07:45 +00:00
|
|
|
|
if (levels[event.level] < levels['error']) {
|
2021-03-17 19:48:49 +00:00
|
|
|
|
dest = process.stdout;
|
|
|
|
|
}
|
|
|
|
|
let type = event.type;
|
2021-03-19 21:07:45 +00:00
|
|
|
|
if (event.level === 'info') {
|
2021-03-17 19:48:49 +00:00
|
|
|
|
type = bold(blue(type));
|
2021-03-19 21:07:45 +00:00
|
|
|
|
} else if (event.level === 'error') {
|
2021-03-17 19:48:49 +00:00
|
|
|
|
type = bold(red(type));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
dest.write(`[${type}] `);
|
2021-03-19 21:07:45 +00:00
|
|
|
|
dest.write(utilFormat(...event.args));
|
2021-03-17 19:48:49 +00:00
|
|
|
|
dest.write('\n');
|
|
|
|
|
|
|
|
|
|
callback();
|
2021-03-19 21:07:45 +00:00
|
|
|
|
},
|
2021-03-17 19:48:49 +00:00
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
interface LogWritable<T> extends Writable {
|
|
|
|
|
write: (chunk: T) => boolean;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
export type LoggerLevel = 'debug' | 'info' | 'warn' | 'error' | 'silent'; // same as Pino
|
|
|
|
|
export type LoggerEvent = 'debug' | 'info' | 'warn' | 'error';
|
|
|
|
|
|
|
|
|
|
export interface LogOptions {
|
|
|
|
|
dest: LogWritable<LogMessage>;
|
2021-03-19 21:07:45 +00:00
|
|
|
|
level: LoggerLevel;
|
2021-03-17 19:48:49 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
export const defaultLogOptions: LogOptions = {
|
|
|
|
|
dest: defaultLogDestination,
|
2021-03-19 21:07:45 +00:00
|
|
|
|
level: 'info',
|
2021-03-17 19:48:49 +00:00
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
export interface LogMessage {
|
|
|
|
|
type: string;
|
2021-03-19 21:07:45 +00:00
|
|
|
|
level: LoggerLevel;
|
2021-03-17 19:48:49 +00:00
|
|
|
|
message: string;
|
2021-03-19 21:07:45 +00:00
|
|
|
|
args: Array<any>;
|
2021-03-17 19:48:49 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const levels: Record<LoggerLevel, number> = {
|
|
|
|
|
debug: 20,
|
|
|
|
|
info: 30,
|
|
|
|
|
warn: 40,
|
|
|
|
|
error: 50,
|
|
|
|
|
silent: 90,
|
|
|
|
|
};
|
|
|
|
|
|
2021-04-01 16:25:28 +00:00
|
|
|
|
/** Full logging API */
|
2021-03-19 21:07:45 +00:00
|
|
|
|
export function log(opts: LogOptions = defaultLogOptions, level: LoggerLevel, type: string, ...args: Array<any>) {
|
|
|
|
|
const event: LogMessage = {
|
2021-03-17 19:48:49 +00:00
|
|
|
|
type,
|
|
|
|
|
level,
|
2021-03-19 21:07:45 +00:00
|
|
|
|
args,
|
|
|
|
|
message: '',
|
2021-03-17 19:48:49 +00:00
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
// test if this level is enabled or not
|
|
|
|
|
if (levels[opts.level] > levels[level]) {
|
|
|
|
|
return; // do nothing
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
opts.dest.write(event);
|
|
|
|
|
}
|
|
|
|
|
|
2021-04-01 16:25:28 +00:00
|
|
|
|
/** Emit a message only shown in debug mode */
|
2021-03-17 19:48:49 +00:00
|
|
|
|
export function debug(opts: LogOptions, type: string, ...messages: Array<any>) {
|
|
|
|
|
return log(opts, 'debug', type, ...messages);
|
|
|
|
|
}
|
|
|
|
|
|
2021-04-01 16:25:28 +00:00
|
|
|
|
/** Emit a general info message (be careful using this too much!) */
|
2021-03-17 19:48:49 +00:00
|
|
|
|
export function info(opts: LogOptions, type: string, ...messages: Array<any>) {
|
|
|
|
|
return log(opts, 'info', type, ...messages);
|
|
|
|
|
}
|
|
|
|
|
|
2021-04-01 16:25:28 +00:00
|
|
|
|
/** Emit a warning a user should be aware of */
|
2021-03-17 19:48:49 +00:00
|
|
|
|
export function warn(opts: LogOptions, type: string, ...messages: Array<any>) {
|
|
|
|
|
return log(opts, 'warn', type, ...messages);
|
|
|
|
|
}
|
|
|
|
|
|
2021-04-01 16:25:28 +00:00
|
|
|
|
/** Emit a fatal error message the user should address. */
|
2021-03-17 19:48:49 +00:00
|
|
|
|
export function error(opts: LogOptions, type: string, ...messages: Array<any>) {
|
|
|
|
|
return log(opts, 'error', type, ...messages);
|
|
|
|
|
}
|
|
|
|
|
|
2021-04-01 16:25:28 +00:00
|
|
|
|
/** Pretty format error for display */
|
2021-03-17 19:48:49 +00:00
|
|
|
|
export function parseError(opts: LogOptions, err: CompileError) {
|
|
|
|
|
let frame = err.frame
|
|
|
|
|
// Switch colons for pipes
|
2021-03-19 21:07:45 +00:00
|
|
|
|
.replace(/^([0-9]+)(:)/gm, `${bold('$1')} │`)
|
2021-03-17 19:48:49 +00:00
|
|
|
|
// Make the caret red.
|
2021-03-19 21:07:45 +00:00
|
|
|
|
.replace(/(?<=^\s+)(\^)/gm, bold(red(' ^')))
|
2021-03-17 19:48:49 +00:00
|
|
|
|
// Add identation
|
2021-03-19 21:07:45 +00:00
|
|
|
|
.replace(/^/gm, ' ');
|
2021-03-17 19:48:49 +00:00
|
|
|
|
|
2021-03-19 21:07:45 +00:00
|
|
|
|
error(
|
|
|
|
|
opts,
|
|
|
|
|
'parse-error',
|
|
|
|
|
`
|
2021-03-17 19:48:49 +00:00
|
|
|
|
|
|
|
|
|
${underline(bold(grey(`${err.filename}:${err.start.line}:${err.start.column}`)))}
|
2021-04-01 16:25:28 +00:00
|
|
|
|
|
2021-03-17 19:48:49 +00:00
|
|
|
|
${bold(red(`𝘅 ${err.message}`))}
|
|
|
|
|
|
|
|
|
|
${frame}
|
2021-03-19 21:07:45 +00:00
|
|
|
|
`
|
|
|
|
|
);
|
2021-03-17 19:48:49 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// A default logger for when too lazy to pass LogOptions around.
|
|
|
|
|
export const logger = {
|
|
|
|
|
debug: debug.bind(null, defaultLogOptions),
|
|
|
|
|
info: info.bind(null, defaultLogOptions),
|
|
|
|
|
warn: warn.bind(null, defaultLogOptions),
|
2021-03-19 21:07:45 +00:00
|
|
|
|
error: error.bind(null, defaultLogOptions),
|
|
|
|
|
};
|