vscode-debugadapter
Version:
Debug adapter implementation for node
103 lines • 12.2 kB
JavaScript
/*---------------------------------------------------------
* Copyright (C) Microsoft Corporation. All rights reserved.
*--------------------------------------------------------*/
Object.defineProperty(exports, "__esModule", { value: true });
exports.trimLastNewline = exports.LogOutputEvent = exports.logger = exports.Logger = exports.LogLevel = void 0;
const internalLogger_1 = require("./internalLogger");
const debugSession_1 = require("./debugSession");
var LogLevel;
(function (LogLevel) {
LogLevel[LogLevel["Verbose"] = 0] = "Verbose";
LogLevel[LogLevel["Log"] = 1] = "Log";
LogLevel[LogLevel["Warn"] = 2] = "Warn";
LogLevel[LogLevel["Error"] = 3] = "Error";
LogLevel[LogLevel["Stop"] = 4] = "Stop";
})(LogLevel = exports.LogLevel || (exports.LogLevel = {}));
class Logger {
constructor() {
this._pendingLogQ = [];
}
log(msg, level = LogLevel.Log) {
msg = msg + '\n';
this._write(msg, level);
}
verbose(msg) {
this.log(msg, LogLevel.Verbose);
}
warn(msg) {
this.log(msg, LogLevel.Warn);
}
error(msg) {
this.log(msg, LogLevel.Error);
}
dispose() {
if (this._currentLogger) {
const disposeP = this._currentLogger.dispose();
this._currentLogger = null;
return disposeP;
}
else {
return Promise.resolve();
}
}
/**
* `log` adds a newline, `write` doesn't
*/
_write(msg, level = LogLevel.Log) {
// [null, undefined] => string
msg = msg + '';
if (this._pendingLogQ) {
this._pendingLogQ.push({ msg, level });
}
else if (this._currentLogger) {
this._currentLogger.log(msg, level);
}
}
/**
* Set the logger's minimum level to log in the console, and whether to log to the file. Log messages are queued before this is
* called the first time, because minLogLevel defaults to Warn.
*/
setup(consoleMinLogLevel, _logFilePath, prependTimestamp = true) {
const logFilePath = typeof _logFilePath === 'string' ?
_logFilePath :
(_logFilePath && this._logFilePathFromInit);
if (this._currentLogger) {
const options = {
consoleMinLogLevel,
logFilePath,
prependTimestamp
};
this._currentLogger.setup(options).then(() => {
// Now that we have a minimum logLevel, we can clear out the queue of pending messages
if (this._pendingLogQ) {
const logQ = this._pendingLogQ;
this._pendingLogQ = null;
logQ.forEach(item => this._write(item.msg, item.level));
}
});
}
}
init(logCallback, logFilePath, logToConsole) {
// Re-init, create new global Logger
this._pendingLogQ = this._pendingLogQ || [];
this._currentLogger = new internalLogger_1.InternalLogger(logCallback, logToConsole);
this._logFilePathFromInit = logFilePath;
}
}
exports.Logger = Logger;
exports.logger = new Logger();
class LogOutputEvent extends debugSession_1.OutputEvent {
constructor(msg, level) {
const category = level === LogLevel.Error ? 'stderr' :
level === LogLevel.Warn ? 'console' :
'stdout';
super(msg, category);
}
}
exports.LogOutputEvent = LogOutputEvent;
function trimLastNewline(str) {
return str.replace(/(\n|\r\n)$/, '');
}
exports.trimLastNewline = trimLastNewline;
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"logger.js","sourceRoot":"","sources":["../src/logger.ts"],"names":[],"mappings":";AAAA;;4DAE4D;;;AAE5D,qDAAkD;AAClD,iDAA6C;AAE7C,IAAY,QAMX;AAND,WAAY,QAAQ;IACnB,6CAAW,CAAA;IACX,qCAAO,CAAA;IACP,uCAAQ,CAAA;IACR,yCAAS,CAAA;IACT,uCAAQ,CAAA;AACT,CAAC,EANW,QAAQ,GAAR,gBAAQ,KAAR,gBAAQ,QAMnB;AA4BD,MAAa,MAAM;IAAnB;QAIS,iBAAY,GAAe,EAAE,CAAC;IA2EvC,CAAC;IAzEA,GAAG,CAAC,GAAW,EAAE,KAAK,GAAG,QAAQ,CAAC,GAAG;QACpC,GAAG,GAAG,GAAG,GAAG,IAAI,CAAC;QACjB,IAAI,CAAC,MAAM,CAAC,GAAG,EAAE,KAAK,CAAC,CAAC;IACzB,CAAC;IAED,OAAO,CAAC,GAAW;QAClB,IAAI,CAAC,GAAG,CAAC,GAAG,EAAE,QAAQ,CAAC,OAAO,CAAC,CAAC;IACjC,CAAC;IAED,IAAI,CAAC,GAAW;QACf,IAAI,CAAC,GAAG,CAAC,GAAG,EAAE,QAAQ,CAAC,IAAI,CAAC,CAAC;IAC9B,CAAC;IAED,KAAK,CAAC,GAAW;QAChB,IAAI,CAAC,GAAG,CAAC,GAAG,EAAE,QAAQ,CAAC,KAAK,CAAC,CAAC;IAC/B,CAAC;IAED,OAAO;QACN,IAAI,IAAI,CAAC,cAAc,EAAE;YACxB,MAAM,QAAQ,GAAG,IAAI,CAAC,cAAc,CAAC,OAAO,EAAE,CAAC;YAC/C,IAAI,CAAC,cAAc,GAAG,IAAI,CAAC;YAC3B,OAAO,QAAQ,CAAC;SAChB;aAAM;YACN,OAAO,OAAO,CAAC,OAAO,EAAE,CAAC;SACzB;IACF,CAAC;IAED;;OAEG;IACK,MAAM,CAAC,GAAW,EAAE,KAAK,GAAG,QAAQ,CAAC,GAAG;QAC/C,8BAA8B;QAC9B,GAAG,GAAG,GAAG,GAAG,EAAE,CAAC;QACf,IAAI,IAAI,CAAC,YAAY,EAAE;YACtB,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,EAAE,GAAG,EAAE,KAAK,EAAE,CAAC,CAAC;SACvC;aAAM,IAAI,IAAI,CAAC,cAAc,EAAE;YAC/B,IAAI,CAAC,cAAc,CAAC,GAAG,CAAC,GAAG,EAAE,KAAK,CAAC,CAAC;SACpC;IACF,CAAC;IAED;;;OAGG;IACH,KAAK,CAAC,kBAA4B,EAAE,YAA6B,EAAE,mBAA4B,IAAI;QAClG,MAAM,WAAW,GAAG,OAAO,YAAY,KAAK,QAAQ,CAAC,CAAC;YACrD,YAAY,CAAC,CAAC;YACd,CAAC,YAAY,IAAI,IAAI,CAAC,oBAAoB,CAAC,CAAC;QAE7C,IAAI,IAAI,CAAC,cAAc,EAAE;YACxB,MAAM,OAAO,GAAG;gBACf,kBAAkB;gBAClB,WAAW;gBACX,gBAAgB;aAChB,CAAC;YACF,IAAI,CAAC,cAAc,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,IAAI,CAAC,GAAG,EAAE;gBAC5C,sFAAsF;gBACtF,IAAI,IAAI,CAAC,YAAY,EAAE;oBACtB,MAAM,IAAI,GAAG,IAAI,CAAC,YAAY,CAAC;oBAC/B,IAAI,CAAC,YAAY,GAAG,IAAI,CAAC;oBACzB,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,EAAE,CAAC,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,GAAG,EAAE,IAAI,CAAC,KAAK,CAAC,CAAC,CAAC;iBACxD;YACF,CAAC,CAAC,CAAC;SAEH;IACF,CAAC;IAED,IAAI,CAAC,WAAyB,EAAE,WAAoB,EAAE,YAAsB;QAC3E,oCAAoC;QACpC,IAAI,CAAC,YAAY,GAAG,IAAI,CAAC,YAAY,IAAI,EAAE,CAAC;QAC5C,IAAI,CAAC,cAAc,GAAG,IAAI,+BAAc,CAAC,WAAW,EAAE,YAAY,CAAC,CAAC;QACpE,IAAI,CAAC,oBAAoB,GAAG,WAAW,CAAC;IACzC,CAAC;CACD;AA/ED,wBA+EC;AAEY,QAAA,MAAM,GAAG,IAAI,MAAM,EAAE,CAAC;AAEnC,MAAa,cAAe,SAAQ,0BAAW;IAC9C,YAAY,GAAW,EAAE,KAAe;QACvC,MAAM,QAAQ,GACb,KAAK,KAAK,QAAQ,CAAC,KAAK,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC;YACrC,KAAK,KAAK,QAAQ,CAAC,IAAI,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC;gBACrC,QAAQ,CAAC;QACV,KAAK,CAAC,GAAG,EAAE,QAAQ,CAAC,CAAC;IACtB,CAAC;CACD;AARD,wCAQC;AAED,SAAgB,eAAe,CAAC,GAAW;IAC1C,OAAO,GAAG,CAAC,OAAO,CAAC,YAAY,EAAE,EAAE,CAAC,CAAC;AACtC,CAAC;AAFD,0CAEC","sourcesContent":["/*---------------------------------------------------------\n * Copyright (C) Microsoft Corporation. All rights reserved.\n *--------------------------------------------------------*/\n\nimport { InternalLogger } from './internalLogger';\nimport { OutputEvent } from './debugSession';\n\nexport enum LogLevel {\n\tVerbose = 0,\n\tLog = 1,\n\tWarn = 2,\n\tError = 3,\n\tStop = 4\n}\n\nexport type ILogCallback = (outputEvent: OutputEvent) => void;\n\ninterface ILogItem {\n\tmsg: string;\n\tlevel: LogLevel;\n}\n\nexport interface ILogger {\n\tlog(msg: string, level?: LogLevel): void;\n\tverbose(msg: string): void;\n\twarn(msg: string): void;\n\terror(msg: string): void;\n}\n\nexport interface IInternalLogger {\n\tdispose(): Promise<void>;\n\tlog(msg: string, level: LogLevel, prependTimestamp?: boolean) : void;\n\tsetup(options: IInternalLoggerOptions): Promise<void>;\n}\n\nexport interface IInternalLoggerOptions {\n\tconsoleMinLogLevel: LogLevel;\n\tlogFilePath?: string;\n\tprependTimestamp?: boolean;\n}\n\nexport class Logger {\n\tprivate _logFilePathFromInit: string;\n\n\tprivate _currentLogger: IInternalLogger;\n\tprivate _pendingLogQ: ILogItem[] = [];\n\n\tlog(msg: string, level = LogLevel.Log): void {\n\t\tmsg = msg + '\\n';\n\t\tthis._write(msg, level);\n\t}\n\n\tverbose(msg: string): void {\n\t\tthis.log(msg, LogLevel.Verbose);\n\t}\n\n\twarn(msg: string): void {\n\t\tthis.log(msg, LogLevel.Warn);\n\t}\n\n\terror(msg: string): void {\n\t\tthis.log(msg, LogLevel.Error);\n\t}\n\n\tdispose(): Promise<void> {\n\t\tif (this._currentLogger) {\n\t\t\tconst disposeP = this._currentLogger.dispose();\n\t\t\tthis._currentLogger = null;\n\t\t\treturn disposeP;\n\t\t} else {\n\t\t\treturn Promise.resolve();\n\t\t}\n\t}\n\n\t/**\n\t * `log` adds a newline, `write` doesn't\n\t */\n\tprivate _write(msg: string, level = LogLevel.Log): void {\n\t\t// [null, undefined] => string\n\t\tmsg = msg + '';\n\t\tif (this._pendingLogQ) {\n\t\t\tthis._pendingLogQ.push({ msg, level });\n\t\t} else if (this._currentLogger) {\n\t\t\tthis._currentLogger.log(msg, level);\n\t\t}\n\t}\n\n\t/**\n\t * Set the logger's minimum level to log in the console, and whether to log to the file. Log messages are queued before this is\n\t * called the first time, because minLogLevel defaults to Warn.\n\t */\n\tsetup(consoleMinLogLevel: LogLevel, _logFilePath?: string|boolean, prependTimestamp: boolean = true): void {\n\t\tconst logFilePath = typeof _logFilePath === 'string' ?\n\t\t\t_logFilePath :\n\t\t\t(_logFilePath && this._logFilePathFromInit);\n\n\t\tif (this._currentLogger) {\n\t\t\tconst options = {\n\t\t\t\tconsoleMinLogLevel,\n\t\t\t\tlogFilePath,\n\t\t\t\tprependTimestamp\n\t\t\t};\n\t\t\tthis._currentLogger.setup(options).then(() => {\n\t\t\t\t// Now that we have a minimum logLevel, we can clear out the queue of pending messages\n\t\t\t\tif (this._pendingLogQ) {\n\t\t\t\t\tconst logQ = this._pendingLogQ;\n\t\t\t\t\tthis._pendingLogQ = null;\n\t\t\t\t\tlogQ.forEach(item => this._write(item.msg, item.level));\n\t\t\t\t}\n\t\t\t});\n\n\t\t}\n\t}\n\n\tinit(logCallback: ILogCallback, logFilePath?: string, logToConsole?: boolean): void {\n\t\t// Re-init, create new global Logger\n\t\tthis._pendingLogQ = this._pendingLogQ || [];\n\t\tthis._currentLogger = new InternalLogger(logCallback, logToConsole);\n\t\tthis._logFilePathFromInit = logFilePath;\n\t}\n}\n\nexport const logger = new Logger();\n\nexport class LogOutputEvent extends OutputEvent {\n\tconstructor(msg: string, level: LogLevel) {\n\t\tconst category =\n\t\t\tlevel === LogLevel.Error ? 'stderr' :\n\t\t\tlevel === LogLevel.Warn ? 'console' :\n\t\t\t'stdout';\n\t\tsuper(msg, category);\n\t}\n}\n\nexport function trimLastNewline(str: string): string {\n\treturn str.replace(/(\\n|\\r\\n)$/, '');\n}\n\n\n"]}
;