Last active
November 25, 2024 03:05
-
-
Save nicholaswmin/a064b36e678b3982e2f5c9192bdc3b7e to your computer and use it in GitHub Desktop.
A pretty-printed SyntaxError for compilers/tokenizers/lexers
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
/* | |
Pretty Syntax Errors | |
> @nicholaswmin, MIT License | |
A SyntaxError that pretty-prints the and higlights error line & column, | |
in your source. | |
This is only useful if you're building a compiler/lexer/tokenizer. | |
Although it is an `instanceof SyntaxError`, their similarities end there. | |
- Isomorphic, pretty prints in Node.js + browser. | |
- Respects `FORCE_COLOR`, `NO_COLOR` env. variables. | |
Additional Notes: | |
- Node.js: The `error.message` is populated with the additional | |
pretty-printing. Includes colors if appropriate. | |
- In browsers, the `err.cause` is used instead. No colors. | |
Browsers immediately call the `message.toString()` method on `new Error()`, | |
instead of waiting until it's actually thrown and that looks awkward. | |
- Safari does not show `error.cause` in it's DevTools so we `console.error` | |
the pretty-printed error on `new Error()`. There's no workaround to this, | |
although it shouldn't realistically be an issue for most use-cases. | |
## Usage: | |
Example: highlight the problematic `3` in `3foo`: | |
``` | |
import { PrettySyntaxError } from './pretty-syntax-error.js' | |
const offset = 41 // position of error in the source string | |
const source = ` // the string you're parsing | |
let exponentiation = => 3 ** 4; | |
let 3foo = 'bar'; | |
let bang_bang_ure_boolean = x => !!x; | |
` | |
throw new PrettySyntaxError('Invalid token', { source, offset }) | |
' Error: SyntaxError ' | |
' ' | |
' Invalid token ' | |
' ⇩ ' | |
'..let 3foo = 'bar'; ' | |
' ⇧ ' | |
' ' | |
' Line: 2 ' | |
' Column: 4 ' | |
``` | |
*/ | |
const envIsBrowser = () => globalThis !== 'undefined' && | |
Object.hasOwn(globalThis, 'window') | |
const browserIsSafari = () => { | |
return envIsBrowser() && | |
globalThis?.window | |
?.navigator?.vendor | |
?.toLowerCase().includes('apple') | |
} | |
class Lines { | |
constructor(message, lines) { | |
this.message = message | |
this.lines = lines | |
} | |
toString() { | |
// Safari doesn't display `err.cause` in its DevTools, | |
// so we gotta log it to the console at least | |
return browserIsSafari() | |
? this.logAndJoin(this.lines) | |
: this.join(this.lines) | |
} | |
join(lines) { | |
return lines.reduce((acc, l) => acc + | |
(this.constructor.canColor() ? | |
l.toANSIString() : | |
l.toString()), '') | |
} | |
logAndJoin(lines) { | |
const joined = this.join(lines) | |
// log in monospace to avoid alignemnt issues | |
globalThis.console.error(`%c${joined}`, 'font-family: monospace;') | |
return joined | |
} | |
// - Normally we'd use `process.stdout.isTTY` but, | |
// theres an issue: https://github.com/nodejs/help/issues/4507 | |
// - Follows guidelines: https://no-color.org/ | |
static canColor() { | |
const defined = v => typeof v !== 'undefined' | |
const IS_TEST = () => defined(process) | |
&& process.env?.NODE_ENV === 'test' | |
const NO_COLOR = () => defined(process.env.NO_COLOR) | |
|| process?.argv?.includes('--no-color') | |
const FORCE_COLOR = () => defined(process.env.FORCE_COLOR) | |
|| process?.argv?.includes('--color') | |
const isTTY = () => !!process.stdout?.isTTY || | |
typeof process.env.NODE_TEST_CONTEXT !== 'undefined' | |
return envIsBrowser() || NO_COLOR() | |
? false : FORCE_COLOR() || isTTY() || IS_TEST() | |
} | |
} | |
class Line { | |
static colors = { | |
'reset': '0', 'red': '31', 'green': '32', 'yellow': '33', | |
'blue': '34', 'magenta': '35', 'cyan': '36', 'white': '37' | |
} | |
constructor(str, color = 'white', offset, { center = false } = {}) { | |
this.str = this.pad(str, offset, { center }) | |
this.color = color | |
} | |
toString() { | |
return this.str | |
} | |
toANSIString() { | |
return `\x1b[${Line.colors[this.color] || '37'}m${this.str}\x1b[0m` | |
} | |
pad(str, offset, { center }) { | |
return ' '.repeat(Math.max(0, offset - (center ? str.length / 2 : 0))) + str | |
} | |
} | |
class Linebreak extends Line { | |
constructor(lines = 1) { | |
super('\n'.repeat(Math.max(1, lines)), 'white', 0) | |
} | |
} | |
class PrettySyntaxError extends SyntaxError { | |
constructor(message, { cause, source, offset }) { | |
const splat = source.slice(0, offset).split('\n') | |
const prior = splat.at(-1).trim() | |
const after = source.slice(offset + 1, source.indexOf('\n', offset)) | |
const error = source.at(offset) | |
const lines = new Lines(message, [ | |
new Linebreak(2), | |
new Line(message, 'red', prior.length, { center: true }), | |
new Linebreak(), | |
new Line('⇩', 'red', prior.length), | |
new Linebreak(), | |
new Line(prior, 'green'), new Line(error, 'red'), new Line(after), | |
new Linebreak(), | |
new Line('⇧', 'red', prior.length), | |
new Linebreak(), | |
new Line(`Line: ${splat.length}`), | |
new Linebreak(), | |
new Line(`Column: ${prior.length + 1}`), | |
new Linebreak() | |
]) | |
super(envIsBrowser() ? message : lines.toString(), { | |
cause: envIsBrowser() ? lines.toString() : 'parsing syntax error' | |
}) | |
this.name = 'SyntaxError' | |
} | |
} | |
export { PrettySyntaxError } |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment
Tests:
no test runner, just
node --test
: