2023-06-22 15:44:26 -04:00
|
|
|
import {Exception, SourceSpan} from 'sass';
|
2023-06-10 12:31:43 -04:00
|
|
|
|
2023-07-02 12:35:17 -04:00
|
|
|
export type ConsoleLogDebug = {
|
2023-06-09 16:59:45 +00:00
|
|
|
options: {
|
|
|
|
span: SourceSpan;
|
|
|
|
};
|
|
|
|
message: string;
|
|
|
|
type: 'debug';
|
|
|
|
};
|
|
|
|
|
2023-07-02 12:35:17 -04:00
|
|
|
export type ConsoleLogWarning = {
|
2023-06-09 16:59:45 +00:00
|
|
|
options: {
|
|
|
|
deprecation: boolean;
|
|
|
|
span?: SourceSpan | undefined;
|
|
|
|
stack?: string | undefined;
|
|
|
|
};
|
|
|
|
message: string;
|
|
|
|
type: 'warn';
|
|
|
|
};
|
2023-06-10 12:31:43 -04:00
|
|
|
|
2023-07-02 12:35:17 -04:00
|
|
|
export type ConsoleLogError = {
|
2023-06-09 16:59:45 +00:00
|
|
|
type: 'error';
|
|
|
|
error: Exception | unknown;
|
|
|
|
};
|
2023-06-10 12:31:43 -04:00
|
|
|
|
2023-06-09 16:59:45 +00:00
|
|
|
export type ConsoleLog = ConsoleLogDebug | ConsoleLogWarning | ConsoleLogError;
|
|
|
|
|
|
|
|
/**
|
2023-06-10 12:31:43 -04:00
|
|
|
* `message` is untrusted.
|
|
|
|
*
|
|
|
|
* Write with `innerText` and then retrieve using `innerHTML` to encode message
|
|
|
|
* for safe display.
|
|
|
|
* @param {string} message The user-submitted string
|
2023-06-09 16:59:45 +00:00
|
|
|
* @return {string} The sanitized string
|
|
|
|
*/
|
2023-06-09 15:35:27 -04:00
|
|
|
function encodeHTML(message: string): string {
|
|
|
|
const el = document.createElement('div');
|
|
|
|
el.innerText = message;
|
|
|
|
return el.innerHTML;
|
2023-06-09 16:59:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
function lineNumberFormatter(number?: number): string {
|
2023-07-02 12:35:17 -04:00
|
|
|
if (number === undefined) return '';
|
2023-06-09 16:59:45 +00:00
|
|
|
number = number + 1;
|
2023-06-10 12:31:43 -04:00
|
|
|
return `${number}`;
|
2023-06-09 16:59:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
export function displayForConsoleLog(item: ConsoleLog): string {
|
2023-06-22 15:44:26 -04:00
|
|
|
const data: {type: string; lineNumber?: number; message: string} = {
|
2023-06-09 16:59:45 +00:00
|
|
|
type: item.type,
|
|
|
|
lineNumber: undefined,
|
|
|
|
message: '',
|
|
|
|
};
|
|
|
|
if (item.type === 'error') {
|
|
|
|
if (item.error instanceof Exception) {
|
|
|
|
data.lineNumber = item.error.span.start.line;
|
|
|
|
}
|
|
|
|
data.message = item.error?.toString() || '';
|
|
|
|
} else if (['debug', 'warn'].includes(item.type)) {
|
|
|
|
data.message = item.message;
|
|
|
|
let lineNumber = item.options.span?.start?.line;
|
|
|
|
if (typeof lineNumber === 'undefined') {
|
|
|
|
const stack = 'stack' in item.options ? item.options.stack : '';
|
|
|
|
const needleFromStackRegex = /^- (\d+):/;
|
|
|
|
const match = stack?.match(needleFromStackRegex);
|
2023-06-10 12:31:43 -04:00
|
|
|
if (match?.[1]) {
|
|
|
|
// Stack trace starts at 1, all others come from span, which starts at
|
|
|
|
// 0, so adjust before formatting.
|
2023-06-09 16:59:45 +00:00
|
|
|
lineNumber = parseInt(match[1]) - 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
data.lineNumber = lineNumber;
|
|
|
|
}
|
|
|
|
|
2023-06-09 15:35:27 -04:00
|
|
|
return `<div class="console-line"><div class="console-location"><span class="console-type console-type-${
|
2023-06-09 16:59:45 +00:00
|
|
|
data.type
|
2023-06-09 15:35:27 -04:00
|
|
|
}">@${data.type}</span>:${lineNumberFormatter(
|
2023-06-22 15:44:26 -04:00
|
|
|
data.lineNumber
|
2023-06-10 12:31:43 -04:00
|
|
|
)}</div><div class="console-message">${encodeHTML(data.message)}</div></div>`;
|
2023-06-09 16:59:45 +00:00
|
|
|
}
|