cordis/logger/logger.ts
2021-11-07 15:57:09 -06:00

260 lines
11 KiB
TypeScript

import * as util from 'util';
import * as path from 'path';
import * as fs from 'fs/promises';
import * as moment from 'moment';
import * as colors from 'colors/safe';
import { SourceMapConsumer } from 'source-map';
import * as StackTrace from '../stack-trace/stack-trace';
const LINE_NUM_LENGTH = 3;
const MAX_FILE_NAME_PADDING = 22;
/** Adds padding to the left of a string */
function padLeft(string: string, padding: string, minLength: number) {
while (string.length < minLength) {
string = padding + string;
}
return string;
}
/** Adds padding to the right of a string */
function padRight(string: string, padding: string, minLength: number) {
while (string.length < minLength) {
string = string + padding;
}
return string;
}
/**
* @param jsFile The compiled javascript file
* @param tsRelativeFile The source typescript file relative to jsFile's directory
* @returns The path to tsRelativeFile based on the root directory
*/
function getPrintedPath(jsFile: string, tsRelativeFile: string) {
let rootFile = path.join(__dirname, '..', '..');
let jsDir = path.dirname(jsFile);
let tsPath = path.resolve(jsDir, tsRelativeFile);
return path.relative(rootFile, tsPath);
}
/**
* @param dir The directory to search
* @returns an async iterator to all files in a directory (recursively searched)
*/
async function* getAllFiles(dir: string): AsyncIterable<string> {
let filesAndDirs = await fs.readdir(dir, { withFileTypes: true });
for (let next of filesAndDirs) {
if (next.isFile()) {
yield path.join(dir, next.name);
} else if (next.isDirectory()) {
let nextFiles = getAllFiles(path.join(dir, next.name));
for await (let nextFile of nextFiles) {
yield nextFile;
}
}
}
}
/**
* @param sourceMaps A mapping from .js file to SourceMapConsumer created from a corresponding .js.map file
* @param frame The StackTrace.Frame to try to convert into typescript format
* @returns A StackTrace.Frame that is in terms of the corresponding typescript file or the same StackTrace.Frame
* if the conversion couldn't be done (i.e. it's an internal node code)
*/
function tryCreateTSFrame(sourceMaps: Map<string, SourceMapConsumer>, frame: StackTrace.Frame): StackTrace.Frame {
if (!frame.fileName) return frame;
let mapping = sourceMaps.get(frame.fileName);
if (!mapping) return frame;
if (!frame.line || !frame.column) return frame;
let pos = mapping.originalPositionFor({ line: frame.line, column: frame.column });
if (!pos.source) return frame;
return {
functionName: frame.functionName,
fileName: path.resolve(path.dirname(frame.fileName), pos.source),
line: pos.line ?? undefined,
column: pos.column ?? undefined
}
}
// I know, I know. But it takes all this disgusting code to make sure we only have to load the source maps once per module instance
// instead of spamming the hard drive. To be honest, why do you care about this file anyway. It's just the logger :)
let STATIC_SOURCE_MAPS_LOADING = false;
let STATIC_SOURCE_MAPS_CALLBACKS: ((sourceMaps: Map<string, SourceMapConsumer>) => void)[] = [];
let STATIC_SOURCE_MAPS: Map<string, SourceMapConsumer> | null = null;
async function getStaticSourceMaps(): Promise<Map<string, SourceMapConsumer>> {
if (STATIC_SOURCE_MAPS !== null) {
return STATIC_SOURCE_MAPS;
} else if (STATIC_SOURCE_MAPS_LOADING) {
return new Promise<Map<string, SourceMapConsumer>>((resolve) => {
STATIC_SOURCE_MAPS_CALLBACKS.push(resolve);
});
} else {
STATIC_SOURCE_MAPS_LOADING = true;
let sourceMaps = new Map<string, SourceMapConsumer>();
for await (let file of getAllFiles(path.join(__dirname, '..'))) {
if (!file.endsWith('.js')) continue;
if (!await fs.stat(file + '.map').then(() => true).catch(() => false)) continue; // make sure .map file exists
let fileBuff = await fs.readFile(file + '.map');
if (typeof window !== 'undefined' && URL && URL.createObjectURL) {
// Only run this stuff if we are running in a browser window
let mappingsWasmBuffer = await fs.readFile(path.join(__dirname, '../../node_modules/source-map/lib/mappings.wasm'));
let mappingsWasmBlob = new Blob([ mappingsWasmBuffer ], { type: 'application/wasm' });
//@ts-ignore Typescript is missing this function... Probably because it is static
SourceMapConsumer.initialize({ "lib/mappings.wasm": URL.createObjectURL(mappingsWasmBlob) });
}
let sourceMapConsumer = await new SourceMapConsumer(fileBuff.toString());
sourceMaps.set(file, sourceMapConsumer);
}
STATIC_SOURCE_MAPS = sourceMaps;
STATIC_SOURCE_MAPS_LOADING = false;
for (let callback of STATIC_SOURCE_MAPS_CALLBACKS) {
callback(sourceMaps);
}
STATIC_SOURCE_MAPS_CALLBACKS = [];
return STATIC_SOURCE_MAPS;
}
}
enum LoggerLevel {
Fatal, Error, Warn,
Info, Debug, Silly
};
/**
* Colorizes text
* @param level The color level
* @param text The text to colorize
* @returns The colored text
*/
function colorize(level: LoggerLevel, text: string): string {
switch (level) {
case LoggerLevel.Fatal: return colors.bgRed(colors.white(text));
case LoggerLevel.Error: return colors.red(text);
case LoggerLevel.Warn: return colors.yellow(text);
case LoggerLevel.Info: return colors.green(text);
case LoggerLevel.Debug: return colors.blue(text);
case LoggerLevel.Silly: return colors.magenta(text);
default: return colors.bgWhite(text);
}
}
export default class Logger {
private name: string;
private console: Console;
private sourceMaps = new Map<string, SourceMapConsumer>();
private hasSourceMaps = false;
private sourceMapsWaiters: (() => void)[] = [];
private longestFileNameLength: number = 0;
private constructor(name: string, processConsole?: Console) {
this.name = name;
this.console = processConsole ?? console;
}
/** Creates a new logger. The logger will print in terms of the dist .js files until it loads the TypeScript mappings */
static create(name: string, processConsole?: Console) {
let log = new Logger(name, processConsole);
(async () => {
let sourceMaps = await getStaticSourceMaps();
log.sourceMaps = sourceMaps;
log.onGetSourceMaps();
})();
return log;
}
private onGetSourceMaps() {
this.hasSourceMaps = true;
for (let sourceMapWaiter of this.sourceMapsWaiters) {
sourceMapWaiter();
}
let longestFileName = Array.from(this.sourceMaps.keys())
.map(filePath => path.basename(filePath))
.reduce((longestFileName, fileName) => fileName.length > longestFileName.length ? fileName : longestFileName, '');
this.longestFileNameLength = longestFileName.length;
}
public async ensureSourceMaps(): Promise<void> {
if (this.hasSourceMaps) return;
return new Promise((resolve) => {
this.sourceMapsWaiters.push(resolve);
});
}
/** Logs a message and potentially corresponding data */
private log(level: LoggerLevel, message: string | null, data?: Error | any): void {
let frames: StackTrace.Frame[] = StackTrace.parse(new Error());
let jsFrame = frames[2];
let tsFrame = tryCreateTSFrame(this.sourceMaps, jsFrame);
let fileName: string;
let line: number;
if (tsFrame.fileName) {
fileName = path.basename(getPrintedPath(jsFrame.fileName, tsFrame.fileName));
line = tsFrame.line ?? NaN;
} else {
fileName = this.name;
line = NaN;
}
let prefixFileName = padLeft(fileName, ' ', Math.min(this.longestFileNameLength, MAX_FILE_NAME_PADDING));
let prefixLine = padRight(line + '', ' ', LINE_NUM_LENGTH);
let prefixLogType = colorize(level, LoggerLevel[level].charAt(0).toLowerCase());
let prefixTime = moment().format('HH:mm:ss.SSS');
let prefix: string = `${prefixTime} ${prefixFileName}:${prefixLine} ${prefixLogType}`;
let out: string = '';
if (message !== null) {
out += message.split('\n').map(o => `${prefix}: ${o}`).join('\n');
}
let handleData = (data: Error | any) => {
if (data) {
if (message !== null) {
out += '\n';
}
if (data instanceof Error) {
if (data.stack) {
const baseDir = path.join(__dirname, '..', '..');
let errorFrames = StackTrace.parse(data);
let tsStackLines = errorFrames
.map((frame: StackTrace.Frame) => tryCreateTSFrame(this.sourceMaps, frame))
.map((frame: StackTrace.Frame) => StackTrace.relativizeFilePath(frame, baseDir))
.map((frame: StackTrace.Frame) => StackTrace.getFrameLine(frame));
out += `${prefix}# ${data.name}: ${data.message}`;
out += `\n${prefix}# ${tsStackLines.join(`\n${prefix}# `)}`;
//out += `${prefix}# ${data.stack.split('\n').join(`\n${prefix}# `)}`;
} else {
out += `${prefix}# ${data.name}: ${data.message}`;
}
// Use the source map to create a typescript version of the stack
// This will be printed asynchronously because SourceMapConsumer
} else {
let s = util.inspect(data, { colors: true, depth: 2 });
s = s.split('\n').map(o => `${prefix}$ ${o}`).join('\n');
out += s;
}
}
}
if (Array.isArray(data)) {
data.forEach(handleData);
} else {
handleData(data);
}
this.console.log(out);
}
public fatal(message: string | null, data?: Error | any): void { this.log(LoggerLevel.Fatal, message, data); }
public error(message: string | null, data?: Error | any): void { this.log(LoggerLevel.Error, message, data); }
public warn( message: string | null, data?: Error | any): void { this.log(LoggerLevel.Warn, message, data); }
public info( message: string | null, data?: Error | any): void { this.log(LoggerLevel.Info, message, data); }
public debug(message: string | null, data?: Error | any): void { this.log(LoggerLevel.Debug, message, data); }
public silly(message: string | null, data?: Error | any): void { this.log(LoggerLevel.Silly, message, data); }
// Wrapper for util.inspect in case we want to do anything special in the future
public inspect(object: any, showHidden?: boolean, depth?: number, color?: boolean): string {
return util.inspect(object, showHidden, depth, color);
}
}