mirror of
https://github.com/oven-sh/bun
synced 2026-02-09 10:28:47 +00:00
3460 lines
135 KiB
JavaScript
3460 lines
135 KiB
JavaScript
import { expect, it, describe } from "bun:test";
|
|
import { hideFromStackTrace } from "harness";
|
|
|
|
describe("Bun.Transpiler", () => {
|
|
const transpiler = new Bun.Transpiler({
|
|
loader: "tsx",
|
|
define: {
|
|
"process.env.NODE_ENV": JSON.stringify("development"),
|
|
user_undefined: "undefined",
|
|
user_nested: "location.origin",
|
|
"hello.earth": "hello.mars",
|
|
"Math.log": "console.error",
|
|
},
|
|
macro: {
|
|
react: {
|
|
bacon: `${import.meta.dir}/macro-check.js`,
|
|
},
|
|
},
|
|
platform: "browser",
|
|
});
|
|
|
|
const ts = {
|
|
parsed: (code, trim = true, autoExport = false) => {
|
|
if (autoExport) {
|
|
code = "export default (" + code + ")";
|
|
}
|
|
|
|
var out = transpiler.transformSync(code, "ts");
|
|
if (autoExport && out.startsWith("export default ")) {
|
|
out = out.substring("export default ".length);
|
|
}
|
|
|
|
if (trim) {
|
|
out = out.trim();
|
|
|
|
if (out.endsWith(";")) {
|
|
out = out.substring(0, out.length - 1);
|
|
}
|
|
|
|
return out.trim();
|
|
}
|
|
|
|
return out;
|
|
},
|
|
|
|
expectPrinted: (code, out) => {
|
|
expect(ts.parsed(code, true, true)).toBe(out);
|
|
},
|
|
|
|
expectPrinted_: (code, out) => {
|
|
expect(ts.parsed(code, !out.endsWith(";\n"), false)).toBe(out);
|
|
},
|
|
|
|
expectParseError: (code, message) => {
|
|
try {
|
|
ts.parsed(code, false, false);
|
|
} catch (er) {
|
|
var err = er;
|
|
if (er instanceof AggregateError) {
|
|
err = er.errors[0];
|
|
}
|
|
expect(err.message).toBe(message);
|
|
|
|
return;
|
|
}
|
|
|
|
throw new Error("Expected parse error for code\n\t" + code);
|
|
},
|
|
};
|
|
hideFromStackTrace(ts.expectPrinted_);
|
|
hideFromStackTrace(ts.expectPrinted);
|
|
hideFromStackTrace(ts.expectParseError);
|
|
|
|
it("normalizes \\r\\n", () => {
|
|
ts.expectPrinted_("console.log(`\r\n\r\n\r\n`)", "console.log(`\n\n\n`);\n");
|
|
});
|
|
|
|
it("doesn't hang indefinitely #2746", () => {
|
|
// this test passes by not hanging
|
|
expect(() =>
|
|
transpiler.transformSync(`
|
|
class Test {
|
|
test() {
|
|
|
|
}
|
|
`),
|
|
).toThrow();
|
|
});
|
|
|
|
describe("property access inlining", () => {
|
|
it("bails out with spread", () => {
|
|
ts.expectPrinted_("const a = [...b][0];", "const a = [...b][0]");
|
|
ts.expectPrinted_("const a = {...b}[0];", "const a = { ...b }[0]");
|
|
});
|
|
it("bails out with multiple items", () => {
|
|
ts.expectPrinted_("const a = [b, c][0];", "const a = [b, c][0]");
|
|
});
|
|
it("works", () => {
|
|
ts.expectPrinted_('const a = ["hey"][0];', 'const a = "hey"');
|
|
});
|
|
it("works nested", () => {
|
|
ts.expectPrinted_('const a = ["hey"][0][0];', 'const a = "h"');
|
|
});
|
|
});
|
|
|
|
describe("TypeScript", () => {
|
|
it("import Foo = Baz.Bar", () => {
|
|
ts.expectPrinted_("import Foo = Baz.Bar;\nexport default Foo;", "const Foo = Baz.Bar;\nexport default Foo");
|
|
});
|
|
|
|
it("ternary should parse correctly when parsing typescript fails", () => {
|
|
ts.expectPrinted_(
|
|
"var c = Math.random() ? ({ ...{} }) : ({ ...{} })",
|
|
"var c = Math.random() ? { ...{} } : { ...{} }",
|
|
);
|
|
});
|
|
|
|
it("should parse empty type parameters", () => {
|
|
const exp = ts.expectPrinted_;
|
|
const err = ts.expectParseError;
|
|
exp("type X<> = never;var x: X", "var x");
|
|
exp("interface X<> {};var x: X", "var x");
|
|
err("class Foo<> {}", 'Expected identifier but found ">"');
|
|
err("function foo<>(): void {}", 'Expected identifier but found ">"');
|
|
err("const x: Foo<> = {}", "Unexpected >");
|
|
});
|
|
|
|
it("should parse infer extends ternary correctly #9959", () => {
|
|
ts.expectPrinted_("type Foo<T> = T extends infer U ? U : never;", "");
|
|
ts.expectPrinted_("var foo: Foo extends string | infer Foo extends string ? Foo : never", "var foo");
|
|
ts.expectPrinted_("var foo: Foo extends string & infer Foo extends string ? Foo : never", "var foo");
|
|
});
|
|
|
|
it.todo("instantiation expressions", async () => {
|
|
const exp = ts.expectPrinted_;
|
|
const err = ts.expectParseError;
|
|
|
|
// TODO: fix Unexpected end of file in these cases
|
|
// exp("f<number>", "f;\n");
|
|
// exp("f<number, boolean>", "f;\n");
|
|
// exp("f.g<number>", "f.g;\n");
|
|
exp("f<number>.g", "f.g;\n");
|
|
// exp("f<number>.g<number>", "f.g;\n");
|
|
// exp("f['g']<number>", 'f["g"];\n');
|
|
// exp("(f<number>)<number>", "f;\n");
|
|
|
|
// Function call
|
|
exp("const x1 = f<true>\n(true);", "const x1 = f(true);\n");
|
|
// Relational expression
|
|
// exp("const x1 = f<true>\ntrue;", "const x1 = f;\ntrue;\n");
|
|
// Instantiation expression
|
|
// exp("const x1 = f<true>;\n(true);", "const x1 = f;\ntrue;\n");
|
|
|
|
// Trailing commas are not allowed
|
|
exp("const x = Array<number>\n(0);", "const x = Array(0);\n");
|
|
// TODO: make these errors
|
|
// exp("const x = Array<number>;\n(0);", "const x = Array;\n0;\n");
|
|
// err("const x = Array<number,>\n(0);", 'Expected identifier but found ">"');
|
|
// err("const x = Array<number,>;\n(0);", 'Expected identifier but found ">"');
|
|
|
|
exp("f<number>?.();", "f?.();\n");
|
|
exp("f?.<number>();", "f?.();\n");
|
|
exp("f<<T>() => T>?.();", "f?.();\n");
|
|
exp("f?.<<T>() => T>();", "f?.();\n");
|
|
|
|
exp("f<number>['g'];", 'f < number > ["g"];\n');
|
|
|
|
exp("type T21 = typeof Array<string>; f();", "f();\n");
|
|
exp("type T22 = typeof Array<string, number>; f();", "f();\n");
|
|
|
|
exp("f<x>, g<y>;", "f, g;\n");
|
|
exp("f<<T>() => T>;", "f;\n");
|
|
exp("f.x<<T>() => T>;", "f.x;\n");
|
|
exp("f['x']<<T>() => T>;", 'f["x"];\n');
|
|
exp("f<x>g<y>;", "f < x > g;\n");
|
|
exp("f<x>=g<y>;", "f = g;\n");
|
|
exp("f<x>>g<y>;", "f < x >> g;\n");
|
|
exp("f<x>>>g<y>;", "f < x >>> g;\n");
|
|
err("f<x>>=g<y>;", "Invalid assignment target");
|
|
err("f<x>>>=g<y>;", "Invalid assignment target");
|
|
exp("f<x> = g<y>;", "f = g;\n");
|
|
err("f<x> > g<y>;", "Unexpected >");
|
|
err("f<x> >> g<y>;", "Unexpected >>");
|
|
err("f<x> >>> g<y>;", "Unexpected >>>");
|
|
err("f<x> >= g<y>;", "Unexpected >=");
|
|
err("f<x> >>= g<y>;", "Unexpected >>=");
|
|
err("f<x> >>>= g<y>;", "Unexpected >>>=");
|
|
exp("a([f<x>]);", "a([f]);\n");
|
|
exp("f<x> ? g<y> : h<z>;", "f ? g : h;\n");
|
|
exp("{ f<x> }", "f");
|
|
exp("f<x> + g<y>;", "f < x > +g;\n");
|
|
exp("f<x> - g<y>;", "f < x > -g;\n");
|
|
exp("f<x> * g<y>;", "f * g;\n");
|
|
exp("f<x> *= g<y>;", "f *= g;\n");
|
|
exp("f<x> == g<y>;", "f == g;\n");
|
|
exp("f<x> ?? g<y>;", "f ?? g;\n");
|
|
exp("f<x> in g<y>;", "f in g;\n");
|
|
exp("f<x> instanceof g<y>;", "f instanceof g;\n");
|
|
exp("f<x> as g<y>;", "f;\n");
|
|
exp("f<x> satisfies g<y>;", "f;\n");
|
|
|
|
err("const a8 = f<number><number>;", "Unexpected ;");
|
|
err("const b1 = f?.<number>;", "Parse error");
|
|
// err("const b1 = f?.<number>;", 'Expected "(" but found ";"');
|
|
|
|
// TODO: why are the JSX tests failing but only in Bun.Transpiler? They
|
|
// work when run manually
|
|
|
|
// See: https://github.com/microsoft/TypeScript/issues/48711
|
|
// exp("type x = y\n<number>\nz", "z;\n");
|
|
// exp("type x = y\n<number>\nz\n</number>", '/* @__PURE__ */ React.createElement("number", null, "z");\n');
|
|
exp("type x = typeof y\n<number>\nz", "z;\n");
|
|
// exp("type x = typeof y\n<number>\nz\n</number>", '/* @__PURE__ */ React.createElement("number", null, "z");\n');
|
|
exp("interface Foo { \n (a: number): a \n <T>(): void \n }", "");
|
|
exp("interface Foo { \n (a: number): a \n <T>(): void \n }", "");
|
|
exp("interface Foo { \n (a: number): typeof a \n <T>(): void \n }", "");
|
|
exp("interface Foo { \n (a: number): typeof a \n <T>(): void \n }", "");
|
|
// err("type x = y\n<number>\nz\n</number>", "Unterminated regular expression");
|
|
// errX(
|
|
// "type x = y\n<number>\nz",
|
|
// 'Unexpected end of file before a closing "number" tag\n<stdin>: NOTE: The opening "number" tag is here:',
|
|
// );
|
|
err("type x = typeof y\n<number>\nz\n</number>", "Syntax Error!!");
|
|
// errX(
|
|
// "type x = typeof y\n<number>\nz",
|
|
// 'Unexpected end of file before a closing "number" tag\n<stdin>: NOTE: The opening "number" tag is here:',
|
|
// );
|
|
|
|
// See: https://github.com/microsoft/TypeScript/issues/48654
|
|
exp("x<true> y", "x < true > y;\n");
|
|
exp("x<true>\ny", "x;\ny;\n");
|
|
exp("x<true>\nif (y) {}", "x;\nif (y)");
|
|
exp("x<true>\nimport 'y'", 'x;\nimport"y";\n');
|
|
exp("x<true>\nimport('y')", 'x;\nimport("y");\n');
|
|
exp("x<true>\na(import.meta)", "x;\na(import.meta);\n");
|
|
exp("x<true> import('y')", 'x < true > import("y");\n');
|
|
exp("x<true> import.meta", "x < true > import.meta;\n");
|
|
exp("new x<number> y", "new x < number > y");
|
|
exp("new x<number>\ny", "new x;\ny");
|
|
exp("new x<number>\nif (y) {}", "new x;\nif (y)");
|
|
exp("new x<true>\nimport 'y'", 'new x;\nimport"y"');
|
|
exp("new x<true>\nimport('y')", 'new x;\nimport("y")');
|
|
exp("new x<true>\na(import.meta)", "new x;\na(import.meta)");
|
|
exp("new x<true> import('y')", 'new x < true > import("y")');
|
|
exp("new x<true> import.meta", "new x < true > import.meta");
|
|
|
|
// See: https://github.com/microsoft/TypeScript/issues/48759
|
|
err("x<true>\nimport<T>('y')", "Unexpected <");
|
|
err("new x<true>\nimport<T>('y')", "Unexpected <");
|
|
|
|
// See: https://github.com/evanw/esbuild/issues/2201
|
|
// err("return Array < ;", "Unexpected ;");
|
|
// err("return Array < > ;", "Unexpected >");
|
|
// err("return Array < , > ;", "Unexpected ,");
|
|
exp("return Array < number > ;", "return Array;\n");
|
|
exp("return Array < number > 1;", "return Array < number > 1;\n");
|
|
exp("return Array < number > +1;", "return Array < number > 1;\n");
|
|
exp("return Array < number > (1);", "return Array(1);\n");
|
|
exp("return Array < number >> 1;", "return Array < number >> 1;\n");
|
|
exp("return Array < number >>> 1;", "return Array < number >>> 1;\n");
|
|
exp("return Array < Array < number >> ;", "return Array;\n");
|
|
exp("return Array < Array < number > > ;", "return Array;\n");
|
|
err("return Array < Array < number > > 1;", "Unexpected >");
|
|
exp("return Array < Array < number >> 1;", "return Array < Array < number >> 1;\n");
|
|
err("return Array < Array < number > > +1;", "Unexpected >");
|
|
exp("return Array < Array < number >> +1;", "return Array < Array < number >> 1;\n");
|
|
exp("return Array < Array < number >> (1);", "return Array(1);\n");
|
|
exp("return Array < Array < number > > (1);", "return Array(1);\n");
|
|
exp("return Array < number > in x;", "return Array in x;\n");
|
|
exp("return Array < Array < number >> in x;", "return Array in x;\n");
|
|
exp("return Array < Array < number > > in x;", "return Array in x;\n");
|
|
exp("for (var x = Array < number > in y) ;", "x = Array;\nfor (x in y)\n ;\nvar x;\n");
|
|
// exp("for (var x = Array < Array < number >> in y) ;", "x = Array;\nfor (var x in y)\n ;\n");
|
|
exp("for (var x = Array < Array < number >> in y) ;", "x = Array;\nfor (x in y)\n ;\nvar x;\n");
|
|
// exp("for (var x = Array < Array < number > > in y) ;", "x = Array;\nfor (var x in y)\n ;\n");
|
|
exp("for (var x = Array < Array < number > > in y) ;", "x = Array;\nfor (x in y)\n ;\nvar x;\n");
|
|
|
|
// See: https://github.com/microsoft/TypeScript/pull/49353
|
|
exp("F<{}> 0", "F < {} > 0;\n");
|
|
exp("F<{}> class F<T> {}", "F < {} > class F {\n};\n");
|
|
exp("f<{}> function f<T>() {}", "f < {} > function f() {\n};\n");
|
|
exp("F<{}>\nabc()", "F;\nabc();\n");
|
|
|
|
// TODO: why are there two newlines here?
|
|
exp("F<{}>()\nclass F<T> {}", "F();\n\nclass F {\n}");
|
|
|
|
exp("f<{}>()\nfunction f<T>() {}", "let f = function() {\n};\nf()");
|
|
});
|
|
|
|
// TODO: fix all the cases that report generic "Parse error"
|
|
it("types", () => {
|
|
const exp = ts.expectPrinted_;
|
|
const err = ts.expectParseError;
|
|
|
|
exp("let x: T extends number\n ? T\n : number", "let x;\n");
|
|
exp("let x: {y: T extends number ? T : number}", "let x;\n");
|
|
exp("let x: {y: T \n extends: number}", "let x;\n");
|
|
exp("let x: {y: T \n extends?: number}", "let x;\n");
|
|
exp("let x: (number | string)[]", "let x;\n");
|
|
exp("let x: [string[]?]", "let x;\n");
|
|
exp("let x: [number?, string?]", "let x;\n");
|
|
exp("let x: [a: number, b?: string, ...c: number[]]", "let x;\n");
|
|
exp("type x =\n A\n | B\n C", "C;\n");
|
|
exp("type x =\n | A\n | B\n C", "C;\n");
|
|
exp("type x =\n A\n & B\n C", "C;\n");
|
|
exp("type x =\n & A\n & B\n C", "C;\n");
|
|
exp("type x = [-1, 0, 1]\na([])", "a([]);\n");
|
|
exp("type x = [-1n, 0n, 1n]\na([])", "a([]);\n");
|
|
exp("type x = {0: number, readonly 1: boolean}\na([])", "a([]);\n");
|
|
exp("type x = {'a': number, readonly 'b': boolean}\na([])", "a([]);\n");
|
|
exp("type\nFoo = {}", "type;\nFoo = {};\n");
|
|
err("export type\nFoo = {}", 'Unexpected newline after "type"');
|
|
exp("let x: {x: 'a', y: false, z: null}", "let x;\n");
|
|
exp("let x: {foo(): void}", "let x;\n");
|
|
exp("let x: {['x']: number}", "let x;\n");
|
|
exp("let x: {['x'](): void}", "let x;\n");
|
|
exp("let x: {[key: string]: number}", "let x;\n");
|
|
exp("let x: {[keyof: string]: number}", "let x;\n");
|
|
exp("let x: {[readonly: string]: number}", "let x;\n");
|
|
exp("let x: {[infer: string]: number}", "let x;\n");
|
|
exp("let x: [keyof: string]", "let x;\n");
|
|
exp("let x: [readonly: string]", "let x;\n");
|
|
exp("let x: [infer: string]", "let x;\n");
|
|
err("let x: A extends B ? keyof : string", "Unexpected :");
|
|
err("let x: A extends B ? readonly : string", "Unexpected :");
|
|
err("let x: A extends B ? infer : string", 'Expected identifier but found ":"');
|
|
err("let x: {[new: string]: number}", 'Expected "(" but found ":"');
|
|
err("let x: {[import: string]: number}", 'Expected "(" but found ":"');
|
|
err("let x: {[typeof: string]: number}", 'Expected identifier but found ":"');
|
|
exp("let x: () => void = Foo", "let x = Foo;\n");
|
|
exp("let x: new () => void = Foo", "let x = Foo;\n");
|
|
exp("let x = 'x' as keyof T", 'let x = "x";\n');
|
|
exp("let x = [1] as readonly [number]", "let x = [1];\n");
|
|
exp("let x = 'x' as keyof typeof Foo", 'let x = "x";\n');
|
|
exp("let fs: typeof import('fs') = require('fs')", 'let fs = require("fs");\n');
|
|
exp("let fs: typeof import('fs').exists = require('fs').exists", 'let fs = require("fs").exists;\n');
|
|
exp("let fs: typeof import('fs', { assert: { type: 'json' } }) = require('fs')", 'let fs = require("fs");\n');
|
|
exp(
|
|
"let fs: typeof import('fs', { assert: { 'resolution-mode': 'import' } }) = require('fs')",
|
|
'let fs = require("fs");\n',
|
|
);
|
|
exp("let x: <T>() => Foo<T>", "let x;\n");
|
|
exp("let x: new <T>() => Foo<T>", "let x;\n");
|
|
exp("let x: <T extends object>() => Foo<T>", "let x;\n");
|
|
exp("let x: new <T extends object>() => Foo<T>", "let x;\n");
|
|
exp("type Foo<T> = {[P in keyof T]?: T[P]}", "");
|
|
exp("type Foo<T> = {[P in keyof T]+?: T[P]}", "");
|
|
exp("type Foo<T> = {[P in keyof T]-?: T[P]}", "");
|
|
exp("type Foo<T> = {readonly [P in keyof T]: T[P]}", "");
|
|
exp("type Foo<T> = {-readonly [P in keyof T]: T[P]}", "");
|
|
exp("type Foo<T> = {+readonly [P in keyof T]: T[P]}", "");
|
|
exp("type Foo<T> = {[infer in T]?: Foo}", "");
|
|
exp("type Foo<T> = {[keyof in T]?: Foo}", "");
|
|
exp("type Foo<T> = {[asserts in T]?: Foo}", "");
|
|
exp("type Foo<T> = {[abstract in T]?: Foo}", "");
|
|
exp("type Foo<T> = {[readonly in T]?: Foo}", "");
|
|
exp("type Foo<T> = {[satisfies in T]?: Foo}", "");
|
|
exp("let x: number! = y", "let x = y;\n");
|
|
// TODO: dead code elimination causes this to become "y;\n" because !
|
|
// operator is side effect free on an identifier
|
|
// exp("let x: number \n !y", "let x;\n!y;\n");
|
|
exp("const x: unique = y", "const x = y;\n");
|
|
exp("const x: unique<T> = y", "const x = y;\n");
|
|
exp("const x: unique\nsymbol = y", "const x = y;\n");
|
|
exp("let x: typeof a = y", "let x = y;\n");
|
|
exp("let x: typeof a.b = y", "let x = y;\n");
|
|
exp("let x: typeof a.if = y", "let x = y;\n");
|
|
exp("let x: typeof if.a = y", "let x = y;\n");
|
|
exp("let x: typeof readonly = y", "let x = y;\n");
|
|
err("let x: typeof readonly Array", 'Expected ";" but found "Array"');
|
|
exp("let x: `y`", "let x;\n");
|
|
err("let x: tag`y`", 'Expected ";" but found "`y`"');
|
|
exp("let x: { <A extends B>(): c.d \n <E extends F>(): g.h }", "let x;\n");
|
|
// exp("type x = a.b \n <c></c>", '/* @__PURE__ */ React.createElement("c", null);\n');
|
|
exp("type Foo = a.b \n | c.d", "");
|
|
exp("type Foo = a.b \n & c.d", "");
|
|
exp("type Foo = \n | a.b \n | c.d", "");
|
|
exp("type Foo = \n & a.b \n & c.d", "");
|
|
exp("type Foo = Bar extends [infer T] ? T : null", "");
|
|
exp("type Foo = Bar extends [infer T extends string] ? T : null", "");
|
|
exp("type Foo = {} extends infer T extends {} ? A<T> : never", "");
|
|
exp("type Foo = {} extends (infer T extends {}) ? A<T> : never", "");
|
|
exp("let x: A extends B<infer C extends D> ? D : never", "let x;\n");
|
|
exp("let x: A extends B<infer C extends D ? infer C : never> ? D : never", "let x;\n");
|
|
exp("let x: ([e1, e2, ...es]: any) => any", "let x;\n");
|
|
exp("let x: (...[e1, e2, es]: any) => any", "let x;\n");
|
|
exp("let x: (...[e1, e2, ...es]: any) => any", "let x;\n");
|
|
exp("let x: (y, [e1, e2, ...es]: any) => any", "let x;\n");
|
|
exp("let x: (y, ...[e1, e2, es]: any) => any", "let x;\n");
|
|
exp("let x: (y, ...[e1, e2, ...es]: any) => any", "let x;\n");
|
|
|
|
exp("let x: A.B<X.Y>", "let x;\n");
|
|
exp("let x: A.B<X.Y>=2", "let x = 2;\n");
|
|
exp("let x: A.B<X.Y<Z>>", "let x;\n");
|
|
exp("let x: A.B<X.Y<Z>>=2", "let x = 2;\n");
|
|
exp("let x: A.B<X.Y<Z<T>>>", "let x;\n");
|
|
exp("let x: A.B<X.Y<Z<T>>>=2", "let x = 2;\n");
|
|
|
|
exp("a((): A<T>=> 0)", "a(() => 0);\n");
|
|
exp("a((): A<B<T>>=> 0)", "a(() => 0);\n");
|
|
exp("a((): A<B<C<T>>>=> 0)", "a(() => 0);\n");
|
|
|
|
exp("let foo: any\n<x>y", "let foo;\ny;\n");
|
|
// expectPrintedTSX(t, "let foo: any\n<x>y</x>", 'let foo;\n/* @__PURE__ */ React.createElement("x", null, "y");\n');
|
|
err("let foo: (any\n<x>y)", 'Expected ")" but found "<"');
|
|
|
|
exp("let foo = bar as (null)", "let foo = bar;\n");
|
|
exp("let foo = bar\nas (null)", "let foo = bar;\nas(null);\n");
|
|
err("let foo = (bar\nas (null))", 'Expected ")" but found "as"');
|
|
|
|
exp("a as any ? b : c;", "a ? b : c;\n");
|
|
exp("a as any ? async () => b : c;", "a || c;\n");
|
|
|
|
exp("foo as number extends Object ? any : any;", "foo;\n");
|
|
exp("foo as number extends Object ? () => void : any;", "foo;\n");
|
|
exp(
|
|
"let a = b ? c : d as T extends T ? T extends T ? T : never : never ? e : f;",
|
|
"let a = b ? c : d ? e : f;\n",
|
|
);
|
|
err("type a = b extends c", 'Expected "?" but found end of file');
|
|
err("type a = b extends c extends d", 'Expected "?" but found "extends"');
|
|
err("type a = b ? c : d", 'Expected ";" but found "?"');
|
|
|
|
exp("let foo: keyof Object = 'toString'", 'let foo = "toString";\n');
|
|
exp("let foo: keyof\nObject = 'toString'", 'let foo = "toString";\n');
|
|
exp("let foo: (keyof\nObject) = 'toString'", 'let foo = "toString";\n');
|
|
|
|
exp("type Foo = Array<<T>(x: T) => T>\n x", "x;\n");
|
|
// expectPrintedTSX(t, "<Foo<<T>(x: T) => T>/>", "/* @__PURE__ */ React.createElement(Foo, null);\n");
|
|
|
|
// Certain built-in types do not accept type parameters
|
|
exp("x as 1 < 1", "x < 1;\n");
|
|
exp("x as 1n < 1", "x < 1;\n");
|
|
exp("x as -1 < 1", "x < 1;\n");
|
|
exp("x as -1n < 1", "x < 1;\n");
|
|
exp("x as '' < 1", "x < 1;\n");
|
|
exp("x as `` < 1", "x < 1;\n");
|
|
exp("x as any < 1", "x < 1;\n");
|
|
exp("x as bigint < 1", "x < 1;\n");
|
|
exp("x as false < 1", "x < 1;\n");
|
|
exp("x as never < 1", "x < 1;\n");
|
|
exp("x as null < 1", "x < 1;\n");
|
|
exp("x as number < 1", "x < 1;\n");
|
|
exp("x as object < 1", "x < 1;\n");
|
|
exp("x as string < 1", "x < 1;\n");
|
|
exp("x as symbol < 1", "x < 1;\n");
|
|
exp("x as this < 1", "x < 1;\n");
|
|
exp("x as true < 1", "x < 1;\n");
|
|
exp("x as undefined < 1", "x < 1;\n");
|
|
exp("x as unique symbol < 1", "x < 1;\n");
|
|
exp("x as unknown < 1", "x < 1;\n");
|
|
exp("x as void < 1", "x < 1;\n");
|
|
err("x as Foo < 1", 'Expected ">" but found end of file');
|
|
|
|
// These keywords are valid tuple labels
|
|
exp("type _false = [false: string]", "");
|
|
exp("type _function = [function: string]", "");
|
|
exp("type _import = [import: string]", "");
|
|
exp("type _new = [new: string]", "");
|
|
exp("type _null = [null: string]", "");
|
|
exp("type _this = [this: string]", "");
|
|
exp("type _true = [true: string]", "");
|
|
exp("type _typeof = [typeof: string]", "");
|
|
exp("type _void = [void: string]", "");
|
|
|
|
// These keywords are invalid tuple labels
|
|
err("type _break = [break: string]", "Unexpected break");
|
|
err("type _case = [case: string]", "Unexpected case");
|
|
err("type _catch = [catch: string]", "Unexpected catch");
|
|
err("type _class = [class: string]", "Unexpected class");
|
|
err("type _const = [const: string]", 'Unexpected "const"');
|
|
err("type _continue = [continue: string]", "Unexpected continue");
|
|
err("type _debugger = [debugger: string]", "Unexpected debugger");
|
|
err("type _default = [default: string]", "Unexpected default");
|
|
err("type _delete = [delete: string]", "Unexpected delete");
|
|
err("type _do = [do: string]", "Unexpected do");
|
|
err("type _else = [else: string]", "Unexpected else");
|
|
err("type _enum = [enum: string]", "Unexpected enum");
|
|
err("type _export = [export: string]", "Unexpected export");
|
|
err("type _extends = [extends: string]", "Unexpected extends");
|
|
err("type _finally = [finally: string]", "Unexpected finally");
|
|
err("type _for = [for: string]", "Unexpected for");
|
|
err("type _if = [if: string]", "Unexpected if");
|
|
err("type _in = [in: string]", "Unexpected in");
|
|
err("type _instanceof = [instanceof: string]", "Unexpected instanceof");
|
|
err("type _return = [return: string]", "Unexpected return");
|
|
err("type _super = [super: string]", "Unexpected super");
|
|
err("type _switch = [switch: string]", "Unexpected switch");
|
|
err("type _throw = [throw: string]", "Unexpected throw");
|
|
err("type _try = [try: string]", "Unexpected try");
|
|
err("type _var = [var: string]", "Unexpected var");
|
|
err("type _while = [while: string]", "Unexpected while");
|
|
err("type _with = [with: string]", "Unexpected with");
|
|
|
|
// TypeScript 4.1
|
|
exp("let foo: `${'a' | 'b'}-${'c' | 'd'}` = 'a-c'", 'let foo = "a-c";\n');
|
|
|
|
// TypeScript 4.2
|
|
exp("let x: abstract new () => void = Foo", "let x = Foo;\n");
|
|
exp("let x: abstract new <T>() => Foo<T>", "let x;\n");
|
|
exp("let x: abstract new <T extends object>() => Foo<T>", "let x;\n");
|
|
err("let x: abstract () => void = Foo", 'Expected ";" but found "("');
|
|
err("let x: abstract <T>() => Foo<T>", 'Expected ";" but found "("');
|
|
err("let x: abstract <T extends object>() => Foo<T>", 'Expected "?" but found ">"');
|
|
|
|
// TypeScript 4.7
|
|
// jsxErrorArrow := "The character \">\" is not valid inside a JSX element\n" +
|
|
// "NOTE: Did you mean to escape it as \"{'>'}\" instead?\n"
|
|
exp("type Foo<in T> = T", "");
|
|
exp("type Foo<out T> = T", "");
|
|
exp("type Foo<in out> = T", "");
|
|
exp("type Foo<out out> = T", "");
|
|
exp("type Foo<in out out> = T", "");
|
|
exp("type Foo<in X, out Y> = [X, Y]", "");
|
|
exp("type Foo<out X, in Y> = [X, Y]", "");
|
|
exp("type Foo<out X, out Y extends keyof X> = [X, Y]", "");
|
|
err("type Foo<i\\u006E T> = T", 'Expected identifier but found "i\\u006E"');
|
|
err("type Foo<ou\\u0074 T> = T", 'Expected ">" but found "T"');
|
|
err("type Foo<in in> = T", 'The modifier "in" is not valid here');
|
|
err("type Foo<out in> = T", 'The modifier "in" is not valid here');
|
|
err("type Foo<out in T> = T", 'The modifier "in" is not valid here');
|
|
err("type Foo<public T> = T", 'Expected ">" but found "T"');
|
|
err("type Foo<in out in T> = T", 'The modifier "in" is not valid here');
|
|
err("type Foo<in out out T> = T", 'The modifier "out" is not valid here');
|
|
exp("class Foo<in T> {}", "class Foo {\n}");
|
|
exp("class Foo<out T> {}", "class Foo {\n}");
|
|
exp("export default class Foo<in T> {}", "export default class Foo {\n}");
|
|
exp("export default class Foo<out T> {}", "export default class Foo {\n}");
|
|
exp("export default class <in T> {}", "export default class {\n}");
|
|
exp("export default class <out T> {}", "export default class {\n}");
|
|
exp("interface Foo<in T> {}", "");
|
|
exp("interface Foo<out T> {}", "");
|
|
exp("declare class Foo<in T> {}", "");
|
|
exp("declare class Foo<out T> {}", "");
|
|
exp("declare interface Foo<in T> {}", "");
|
|
exp("declare interface Foo<out T> {}", "");
|
|
err("function foo<in T>() {}", 'The modifier "in" is not valid here');
|
|
err("function foo<out T>() {}", 'The modifier "out" is not valid here');
|
|
err("export default function foo<in T>() {}", 'The modifier "in" is not valid here');
|
|
err("export default function foo<out T>() {}", 'The modifier "out" is not valid here');
|
|
err("export default function <in T>() {}", 'The modifier "in" is not valid here');
|
|
err("export default function <out T>() {}", 'The modifier "out" is not valid here');
|
|
err("let foo: Foo<in T>", "Unexpected in");
|
|
err("let foo: Foo<out T>", 'Expected ">" but found "T"');
|
|
err("declare function foo<in T>()", 'The modifier "in" is not valid here');
|
|
err("declare function foo<out T>()", 'The modifier "out" is not valid here');
|
|
err("declare let foo: Foo<in T>", "Unexpected in");
|
|
err("declare let foo: Foo<out T>", 'Expected ">" but found "T"');
|
|
exp("Foo = class <in T> {}", "Foo = class {\n}");
|
|
exp("Foo = class <out T> {}", "Foo = class {\n}");
|
|
exp("Foo = class Bar<in T> {}", "Foo = class Bar {\n}");
|
|
exp("Foo = class Bar<out T> {}", "Foo = class Bar {\n}");
|
|
err("foo = function <in T>() {}", 'The modifier "in" is not valid here');
|
|
err("foo = function <out T>() {}", 'The modifier "out" is not valid here');
|
|
err("class Foo { foo<in T>(): T {} }", 'The modifier "in" is not valid here');
|
|
err("class Foo { foo<out T>(): T {} }", 'The modifier "out" is not valid here');
|
|
err("foo = { foo<in T>(): T {} }", 'The modifier "in" is not valid here');
|
|
err("foo = { foo<out T>(): T {} }", 'The modifier "out" is not valid here');
|
|
err("<in T>() => {}", 'The modifier "in" is not valid here');
|
|
err("<out T>() => {}", 'The modifier "out" is not valid here');
|
|
err("<in T, out T>() => {}", 'The modifier "in" is not valid here');
|
|
err("let x: <in T>() => {}", 'The modifier "in" is not valid here');
|
|
err("let x: <out T>() => {}", 'The modifier "out" is not valid here');
|
|
err("let x: <in T, out T>() => {}", 'The modifier "in" is not valid here');
|
|
err("let x: new <in T>() => {}", 'The modifier "in" is not valid here');
|
|
err("let x: new <out T>() => {}", 'The modifier "out" is not valid here');
|
|
err("let x: new <in T, out T>() => {}", 'The modifier "in" is not valid here');
|
|
|
|
err("let x: { y<in T>(): any }", 'The modifier "in" is not valid here');
|
|
err("let x: { y<out T>(): any }", 'The modifier "out" is not valid here');
|
|
err("let x: new <in T, out T>() => {}", 'The modifier "in" is not valid here');
|
|
|
|
// expectPrintedTSX(t, "<in T></in>", "/* @__PURE__ */ React.createElement(\"in\", { T: true });\n")
|
|
// expectPrintedTSX(t, "<out T></out>", "/* @__PURE__ */ React.createElement(\"out\", { T: true });\n")
|
|
// expectPrintedTSX(t, "<in out T></in>", "/* @__PURE__ */ React.createElement(\"in\", { out: true, T: true });\n")
|
|
// expectPrintedTSX(t, "<out in T></out>", "/* @__PURE__ */ React.createElement(\"out\", { in: true, T: true });\n")
|
|
// expectPrintedTSX(t, "<in T extends={true}></in>", "/* @__PURE__ */ React.createElement(\"in\", { T: true, extends: true });\n")
|
|
// expectPrintedTSX(t, "<out T extends={true}></out>", "/* @__PURE__ */ React.createElement(\"out\", { T: true, extends: true });\n")
|
|
// expectPrintedTSX(t, "<in out T extends={true}></in>", "/* @__PURE__ */ React.createElement(\"in\", { out: true, T: true, extends: true });\n")
|
|
// errX(t, "<in T,>() => {}", "Expected \">\" but found \",\"\n")
|
|
// errX(t, "<out T,>() => {}", "Expected \">\" but found \",\"\n")
|
|
// errX(t, "<in out T,>() => {}", "Expected \">\" but found \",\"\n")
|
|
// errX(t, "<in T extends any>() => {}", jsxErrorArrow+"Unexpected end of file before a closing \"in\" tag\n<stdin>: NOTE: The opening \"in\" tag is here:\n")
|
|
// errX(t, "<out T extends any>() => {}", jsxErrorArrow+"Unexpected end of file before a closing \"out\" tag\n<stdin>: NOTE: The opening \"out\" tag is here:\n")
|
|
// errX(t, "<in out T extends any>() => {}", jsxErrorArrow+"Unexpected end of file before a closing \"in\" tag\n<stdin>: NOTE: The opening \"in\" tag is here:\n")
|
|
exp("class Container { get data(): typeof this.#data {} }", "class Container {\n get data() {\n }\n}");
|
|
exp("const a: typeof this.#a = 1;", "const a = 1;\n");
|
|
err("const a: typeof #a = 1;", 'Expected identifier but found "#a"');
|
|
|
|
// TypeScript 5.0
|
|
exp("class Foo<const T> {}", "class Foo {\n}");
|
|
exp("class Foo<const T extends X> {}", "class Foo {\n}");
|
|
exp("Foo = class <const T> {}", "Foo = class {\n}");
|
|
exp("Foo = class Bar<const T> {}", "Foo = class Bar {\n}");
|
|
exp("function foo<const T>() {}", "let foo = function() {\n}");
|
|
exp("foo = function <const T>() {}", "foo = function() {\n}");
|
|
exp("foo = function bar<const T>() {}", "foo = function bar() {\n}");
|
|
exp("class Foo { bar<const T>() {} }", "class Foo {\n bar() {\n }\n}");
|
|
exp("interface Foo { bar<const T>(): T }", "");
|
|
exp("interface Foo { new bar<const T>(): T }", "");
|
|
exp("let x: { bar<const T>(): T }", "let x;\n");
|
|
exp("let x: { new bar<const T>(): T }", "let x;\n");
|
|
exp("foo = { bar<const T>() {} }", "foo = { bar() {\n} }");
|
|
exp("x = <const>(y)", "x = y;\n");
|
|
exp("export let f = <const T>() => {}", "export let f = () => {\n};\n");
|
|
exp("export let f = <const const T>() => {}", "export let f = () => {\n};\n");
|
|
exp("export let f = async <const T>() => {}", "export let f = async () => {\n};\n");
|
|
exp("export let f = async <const const T>() => {}", "export let f = async () => {\n};\n");
|
|
exp("let x: <const T>() => T = y", "let x = y;\n");
|
|
exp("let x: <const const T>() => T = y", "let x = y;\n");
|
|
exp("let x: new <const T>() => T = y", "let x = y;\n");
|
|
exp("let x: new <const const T>() => T = y", "let x = y;\n");
|
|
err("type Foo<const T> = T", 'The modifier "const" is not valid here');
|
|
err("interface Foo<const T> {}", 'The modifier "const" is not valid here');
|
|
err("let x: <const>() => {}", 'Expected identifier but found ">"');
|
|
err("let x: new <const>() => {}", 'Expected identifier but found ">"');
|
|
err("let x: Foo<const T>", 'Expected ">" but found "T"');
|
|
err("x = <T,>(y)", 'Expected "=>" but found end of file');
|
|
err("x = <const T>(y)", 'Expected "=>" but found end of file');
|
|
err("x = <T extends X>(y)", 'Expected "=>" but found end of file');
|
|
err("x = async <T,>(y)", 'Expected "=>" but found end of file');
|
|
err("x = async <const T>(y)", 'Expected "=>" but found end of file');
|
|
err("x = async <T extends X>(y)", 'Expected "=>" but found end of file');
|
|
err("x = <const const>() => {}", 'Expected ">" but found "const"');
|
|
exp("class Foo<const const const T> {}", "class Foo {\n}");
|
|
exp("class Foo<const in out T> {}", "class Foo {\n}");
|
|
exp("class Foo<in const out T> {}", "class Foo {\n}");
|
|
exp("class Foo<in out const T> {}", "class Foo {\n}");
|
|
exp("class Foo<const in const out const T> {}", "class Foo {\n}");
|
|
err("class Foo<in const> {}", 'Expected identifier but found ">"');
|
|
err("class Foo<out const> {}", 'Expected identifier but found ">"');
|
|
err("class Foo<in out const> {}", 'Expected identifier but found ">"');
|
|
// expectPrintedTSX(t, "<const>(x)</const>", '/* @__PURE__ */ React.createElement("const", null, "(x)");\n');
|
|
// expectPrintedTSX(t, "<const const/>", '/* @__PURE__ */ React.createElement("const", { const: true });\n');
|
|
// expectPrintedTSX(t, "<const const></const>", '/* @__PURE__ */ React.createElement("const", { const: true });\n');
|
|
// expectPrintedTSX(t, "<const T/>", '/* @__PURE__ */ React.createElement("const", { T: true });\n');
|
|
// expectPrintedTSX(t, "<const T></const>", '/* @__PURE__ */ React.createElement("const", { T: true });\n');
|
|
// expectPrintedTSX(
|
|
// t,
|
|
// "<const T>(y) = {}</const>",
|
|
// '/* @__PURE__ */ React.createElement("const", { T: true }, "(y) = ");\n',
|
|
// );
|
|
// expectPrintedTSX(
|
|
// t,
|
|
// "<const T extends/>",
|
|
// '/* @__PURE__ */ React.createElement("const", { T: true, extends: true });\n',
|
|
// );
|
|
// expectPrintedTSX(
|
|
// t,
|
|
// "<const T extends></const>",
|
|
// '/* @__PURE__ */ React.createElement("const", { T: true, extends: true });\n',
|
|
// );
|
|
// expectPrintedTSX(
|
|
// t,
|
|
// "<const T extends>(y) = {}</const>",
|
|
// '/* @__PURE__ */ React.createElement("const", { T: true, extends: true }, "(y) = ");\n',
|
|
// );
|
|
// expectPrintedTSX(t, "<const T,>() => {}", "() => {\n};\n");
|
|
// expectPrintedTSX(t, "<const T, X>() => {}", "() => {\n};\n");
|
|
// expectPrintedTSX(t, "<const T, const X>() => {}", "() => {\n};\n");
|
|
// expectPrintedTSX(t, "<const T, const const X>() => {}", "() => {\n};\n");
|
|
// expectPrintedTSX(t, "<const T extends X>() => {}", "() => {\n};\n");
|
|
// expectPrintedTSX(t, "async <const T,>() => {}", "async () => {\n};\n");
|
|
// expectPrintedTSX(t, "async <const T, X>() => {}", "async () => {\n};\n");
|
|
// expectPrintedTSX(t, "async <const T, const X>() => {}", "async () => {\n};\n");
|
|
// expectPrintedTSX(t, "async <const T, const const X>() => {}", "async () => {\n};\n");
|
|
// expectPrintedTSX(t, "async <const T extends X>() => {}", "async () => {\n};\n");
|
|
// errX(
|
|
// t,
|
|
// "<const T>() => {}",
|
|
// jsxErrorArrow +
|
|
// 'Unexpected end of file before a closing "const" tag\n<stdin>: NOTE: The opening "const" tag is here:\n',
|
|
// );
|
|
// errX(
|
|
// t,
|
|
// "<const const>() => {}",
|
|
// jsxErrorArrow +
|
|
// 'Unexpected end of file before a closing "const" tag\n<stdin>: NOTE: The opening "const" tag is here:\n',
|
|
// );
|
|
// errX(t, "<const const T,>() => {}", 'Expected ">" but found ","');
|
|
// errX(
|
|
// t,
|
|
// "<const const T extends X>() => {}",
|
|
// jsxErrorArrow +
|
|
// 'Unexpected end of file before a closing "const" tag\n<stdin>: NOTE: The opening "const" tag is here:\n',
|
|
// );
|
|
err("async <const T>() => {}", "Unexpected const");
|
|
err("async <const const>() => {}", "Unexpected const");
|
|
|
|
// TODO: why doesn't this one fail?
|
|
// err("async <const const T,>() => {}", "Unexpected const");
|
|
// err("async <const const T extends X>() => {}", "Unexpected const");
|
|
});
|
|
|
|
it("modifiers", () => {
|
|
const exp = ts.expectPrinted_;
|
|
|
|
exp("class Foo { public foo: number }", "class Foo {\n foo;\n}");
|
|
exp("class Foo { private foo: number }", "class Foo {\n foo;\n}");
|
|
exp("class Foo { protected foo: number }", "class Foo {\n foo;\n}");
|
|
exp("class Foo { declare foo: number }", "class Foo {\n}");
|
|
exp("class Foo { declare public foo: number }", "class Foo {\n}");
|
|
exp("class Foo { public declare foo: number }", "class Foo {\n}");
|
|
exp("class Foo { override foo: number }", "class Foo {\n foo;\n}");
|
|
exp("class Foo { override public foo: number }", "class Foo {\n foo;\n}");
|
|
exp("class Foo { public override foo: number }", "class Foo {\n foo;\n}");
|
|
exp("class Foo { declare override public foo: number }", "class Foo {\n}");
|
|
exp("class Foo { declare foo = 123 }", "class Foo {\n}");
|
|
|
|
exp("class Foo { public static foo: number }", "class Foo {\n static foo;\n}");
|
|
exp("class Foo { private static foo: number }", "class Foo {\n static foo;\n}");
|
|
exp("class Foo { protected static foo: number }", "class Foo {\n static foo;\n}");
|
|
exp("class Foo { declare static foo: number }", "class Foo {\n}");
|
|
exp("class Foo { declare public static foo: number }", "class Foo {\n}");
|
|
exp("class Foo { public declare static foo: number }", "class Foo {\n}");
|
|
exp("class Foo { public static declare foo: number }", "class Foo {\n}");
|
|
exp("class Foo { override static foo: number }", "class Foo {\n static foo;\n}");
|
|
exp("class Foo { override public static foo: number }", "class Foo {\n static foo;\n}");
|
|
exp("class Foo { public override static foo: number }", "class Foo {\n static foo;\n}");
|
|
exp("class Foo { public static override foo: number }", "class Foo {\n static foo;\n}");
|
|
exp("class Foo { declare override public static foo: number }", "class Foo {\n}");
|
|
exp("class Foo { declare static foo = 123 }", "class Foo {\n}");
|
|
exp("class Foo { static declare foo = 123 }", "class Foo {\n}");
|
|
|
|
exp("let x: abstract new () => void = Foo", "let x = Foo");
|
|
exp("let x: abstract new <T>() => Foo<T>", "let x");
|
|
});
|
|
|
|
it("as", () => {
|
|
const exp = ts.expectPrinted_;
|
|
exp("x as 1 < 1", "x < 1");
|
|
exp("x as 1n < 1", "x < 1");
|
|
exp("x as -1 < 1", "x < 1");
|
|
exp("x as -1n < 1", "x < 1");
|
|
exp("x as '' < 1", "x < 1");
|
|
exp("x as `` < 1", "x < 1");
|
|
exp("x as any < 1", "x < 1");
|
|
exp("x as bigint < 1", "x < 1");
|
|
exp("x as false < 1", "x < 1");
|
|
exp("x as never < 1", "x < 1");
|
|
exp("x as null < 1", "x < 1");
|
|
exp("x as number < 1", "x < 1");
|
|
exp("x as object < 1", "x < 1");
|
|
exp("x as string < 1", "x < 1");
|
|
exp("x as symbol < 1", "x < 1");
|
|
exp("x as this < 1", "x < 1");
|
|
exp("x as true < 1", "x < 1");
|
|
exp("x as undefined < 1", "x < 1");
|
|
exp("x as unique symbol < 1", "x < 1");
|
|
exp("x as unknown < 1", "x < 1");
|
|
exp("x as void < 1", "x < 1");
|
|
});
|
|
|
|
it("class constructor", () => {
|
|
const fixtures = [
|
|
[
|
|
`class Test {
|
|
b: string;
|
|
|
|
constructor(private a: string) {
|
|
this.b = a;
|
|
}
|
|
}`,
|
|
|
|
`
|
|
class Test {
|
|
a;
|
|
b;
|
|
constructor(a) {
|
|
this.a = a;
|
|
this.b = a;
|
|
}
|
|
}
|
|
`.trim(),
|
|
],
|
|
[
|
|
`class Test extends Bar {
|
|
b: string;
|
|
|
|
constructor(private a: string) {
|
|
super();
|
|
this.b = a;
|
|
}
|
|
}`,
|
|
|
|
`
|
|
class Test extends Bar {
|
|
a;
|
|
b;
|
|
constructor(a) {
|
|
super();
|
|
this.a = a;
|
|
this.b = a;
|
|
}
|
|
}
|
|
`.trim(),
|
|
],
|
|
];
|
|
|
|
for (const [code, out] of fixtures) {
|
|
expect(ts.parsed(code, false, false).trim()).toBe(out);
|
|
expect(
|
|
ts
|
|
.parsed("var Test = " + code.trim(), false, false)
|
|
.trim()
|
|
.replaceAll("\n", "")
|
|
.replaceAll(" ", ""),
|
|
).toBe(("var Test = " + out.trim() + ";\n").replaceAll("\n", "").replaceAll(" ", ""));
|
|
}
|
|
});
|
|
|
|
it("import Foo = require('bar')", () => {
|
|
ts.expectPrinted_("import React = require('react')", 'const React = require("react")');
|
|
});
|
|
|
|
it("import type Foo = require('bar')", () => {
|
|
ts.expectPrinted_("import type Foo = require('bar')", "");
|
|
});
|
|
|
|
it("unused import = gets removed", () => {
|
|
ts.expectPrinted_("import Foo = Baz.Bar;", "");
|
|
});
|
|
|
|
it("export import Foo = Baz.Bar", () => {
|
|
ts.expectPrinted_("export import Foo = Baz.Bar;", "export const Foo = Baz.Bar");
|
|
});
|
|
|
|
it("export = {foo: 123}", () => {
|
|
ts.expectPrinted_("export = {foo: 123}", "module.exports = { foo: 123 }");
|
|
});
|
|
|
|
it("export default class implements TypeScript regression", () => {
|
|
expect(
|
|
transpiler
|
|
.transformSync(
|
|
`
|
|
export default class implements ITest {
|
|
async* runTest(path: string): AsyncGenerator<number> {
|
|
yield Math.random();
|
|
}
|
|
}
|
|
`,
|
|
"ts",
|
|
)
|
|
.trim(),
|
|
).toBe(
|
|
`
|
|
export default class {
|
|
async* runTest(path) {
|
|
yield Math.random();
|
|
}
|
|
}
|
|
`.trim(),
|
|
);
|
|
});
|
|
|
|
it("satisfies", () => {
|
|
ts.expectPrinted_("const t1 = { a: 1 } satisfies I1;", "const t1 = { a: 1 };\n");
|
|
ts.expectPrinted_("const t2 = { a: 1, b: 1 } satisfies I1;", "const t2 = { a: 1, b: 1 };\n");
|
|
ts.expectPrinted_("const t3 = { } satisfies I1;", "const t3 = {};\n");
|
|
ts.expectPrinted_("const t4: T1 = { a: 'a' } satisfies T1;", 'const t4 = { a: "a" };\n');
|
|
ts.expectPrinted_("const t5 = (m => m.substring(0)) satisfies T2;", "const t5 = (m) => m.substring(0);\n");
|
|
ts.expectPrinted_("const t6 = [1, 2] satisfies [number, number];", "const t6 = [1, 2];\n");
|
|
ts.expectPrinted_("let t7 = { a: 'test' } satisfies A;", 'let t7 = { a: "test" };\n');
|
|
ts.expectPrinted_("let t8 = { a: 'test', b: 'test' } satisfies A;", 'let t8 = { a: "test", b: "test" };\n');
|
|
ts.expectPrinted_("export default {} satisfies Foo;", "export default {};\n");
|
|
ts.expectPrinted_("export default { a: 1 } satisfies Foo;", "export default { a: 1 };\n");
|
|
ts.expectPrinted_(
|
|
"const p = { isEven: n => n % 2 === 0, isOdd: n => n % 2 === 1 } satisfies Predicates;",
|
|
"const p = { isEven: (n) => n % 2 === 0, isOdd: (n) => n % 2 === 1 };\n",
|
|
);
|
|
ts.expectPrinted_(
|
|
"let obj: { f(s: string): void } & Record<string, unknown> = { f(s) { }, g(s) { } } satisfies { g(s: string): void } & Record<string, unknown>;",
|
|
"let obj = { f(s) {\n}, g(s) {\n} };\n",
|
|
);
|
|
ts.expectPrinted_(
|
|
"const car = { start() { }, move(d) { }, stop() { } } satisfies Movable & Record<string, unknown>;",
|
|
"const car = { start() {\n}, move(d) {\n}, stop() {\n} };\n",
|
|
);
|
|
ts.expectPrinted_("var v = undefined satisfies 1;", "var v = undefined;\n");
|
|
ts.expectPrinted_("const a = { x: 10 } satisfies Partial<Point2d>;", "const a = { x: 10 };\n");
|
|
ts.expectPrinted_(
|
|
'const p = { a: 0, b: "hello", x: 8 } satisfies Partial<Record<Keys, unknown>>;',
|
|
'const p = { a: 0, b: "hello", x: 8 };\n',
|
|
);
|
|
ts.expectPrinted_(
|
|
'const p = { a: 0, b: "hello", x: 8 } satisfies Record<Keys, unknown>;',
|
|
'const p = { a: 0, b: "hello", x: 8 };\n',
|
|
);
|
|
ts.expectPrinted_('const x2 = { m: true, s: "false" } satisfies Facts;', 'const x2 = { m: true, s: "false" };\n');
|
|
ts.expectPrinted_(
|
|
"export const Palette = { white: { r: 255, g: 255, b: 255 }, black: { r: 0, g: 0, d: 0 }, blue: { r: 0, g: 0, b: 255 }, } satisfies Record<string, Color>;",
|
|
"export const Palette = { white: { r: 255, g: 255, b: 255 }, black: { r: 0, g: 0, d: 0 }, blue: { r: 0, g: 0, b: 255 } };\n",
|
|
);
|
|
ts.expectPrinted_('const a: "baz" = "foo" satisfies "foo" | "bar";', 'const a = "foo";\n');
|
|
ts.expectPrinted_(
|
|
'const b: { xyz: "baz" } = { xyz: "foo" } satisfies { xyz: "foo" | "bar" };',
|
|
'const b = { xyz: "foo" };\n',
|
|
);
|
|
});
|
|
});
|
|
|
|
describe("generated closures", () => {
|
|
const input1 = `namespace test {
|
|
export enum x { y }
|
|
}`;
|
|
const output1 = `var test;
|
|
(function(test) {
|
|
let x;
|
|
(function(x) {
|
|
x[x["y"] = 0] = "y";
|
|
})(x = test.x || (test.x = {}));
|
|
})(test || (test = {}))`;
|
|
|
|
it("namespace with exported enum", () => {
|
|
ts.expectPrinted_(input1, output1);
|
|
});
|
|
|
|
const input2 = `export namespace test {
|
|
export enum x { y }
|
|
}`;
|
|
const output2 = `export var test;
|
|
(function(test) {
|
|
let x;
|
|
(function(x) {
|
|
x[x["y"] = 0] = "y";
|
|
})(x = test.x || (test.x = {}));
|
|
})(test || (test = {}))`;
|
|
|
|
it("exported namespace with exported enum", () => {
|
|
ts.expectPrinted_(input2, output2);
|
|
});
|
|
|
|
const input3 = `namespace first {
|
|
export namespace second {
|
|
enum x { y }
|
|
}
|
|
}`;
|
|
const output3 = `var first;
|
|
(function(first) {
|
|
let second;
|
|
(function(second) {
|
|
let x;
|
|
(function(x) {
|
|
x[x["y"] = 0] = "y";
|
|
})(x || (x = {}));
|
|
})(second = first.second || (first.second = {}));
|
|
})(first || (first = {}))`;
|
|
|
|
it("exported inner namespace", () => {
|
|
ts.expectPrinted_(input3, output3);
|
|
});
|
|
|
|
const input4 = `export enum x { y }`;
|
|
const output4 = `export var x;
|
|
(function(x) {
|
|
x[x["y"] = 0] = "y";
|
|
})(x || (x = {}))`;
|
|
|
|
it("exported enum", () => {
|
|
ts.expectPrinted_(input4, output4);
|
|
});
|
|
});
|
|
|
|
describe("exports.replace", () => {
|
|
const transpiler = new Bun.Transpiler({
|
|
exports: {
|
|
replace: {
|
|
// export var foo = function() { }
|
|
// =>
|
|
// export var foo = "bar";
|
|
foo: "bar",
|
|
|
|
// export const getStaticProps = /* code */
|
|
// =>
|
|
// export var __N_SSG = true;
|
|
getStaticProps: ["__N_SSG", true],
|
|
getStaticPaths: ["__N_SSG", true],
|
|
// export function getStaticProps(ctx) { /* code */ }
|
|
// =>
|
|
// export var __N_SSP = true;
|
|
getServerSideProps: ["__N_SSP", true],
|
|
},
|
|
|
|
// Explicitly remove the top-level export, even if it is in use by
|
|
// another part of the file
|
|
eliminate: ["loader", "localVarToRemove"],
|
|
},
|
|
/* only per-file for now, so this isn't good yet */
|
|
treeShaking: true,
|
|
|
|
// remove non-bare unused exports, even if they may have side effects
|
|
// Consistent with tsc & esbuild, this is enabled by default for TypeScript files
|
|
// this flag lets you enable it for JavaScript files
|
|
// this already existed, just wasn't exposed in the API
|
|
trimUnusedImports: true,
|
|
});
|
|
|
|
it("a deletes dead exports and any imports only referenced in dead regions", () => {
|
|
const out = transpiler.transformSync(`
|
|
import {getUserById} from './my-database';
|
|
|
|
export async function getStaticProps(ctx){
|
|
return { props: { user: await getUserById(ctx.params.id) } };
|
|
}
|
|
|
|
export default function MyComponent({user}) {
|
|
getStaticProps();
|
|
return <div id='user'>{user.name}</div>;
|
|
}
|
|
`);
|
|
});
|
|
|
|
it("deletes dead exports and any imports only referenced in dead regions", () => {
|
|
const output = transpiler.transformSync(`
|
|
import deadFS from 'fs';
|
|
import liveFS from 'fs';
|
|
|
|
export var deleteMe = 100;
|
|
|
|
export function loader() {
|
|
deadFS.readFileSync("/etc/passwd");
|
|
liveFS.readFileSync("/etc/passwd");
|
|
}
|
|
|
|
export function action() {
|
|
require("foo");
|
|
liveFS.readFileSync("/etc/passwd")
|
|
deleteMe = 101;
|
|
}
|
|
|
|
export function baz() {
|
|
require("bar");
|
|
}
|
|
`);
|
|
expect(output.includes("loader")).toBe(false);
|
|
expect(output.includes("react")).toBe(false);
|
|
expect(output.includes("action")).toBe(true);
|
|
expect(output.includes("deadFS")).toBe(false);
|
|
expect(output.includes("liveFS")).toBe(true);
|
|
});
|
|
|
|
it.todo("supports replacing exports", () => {
|
|
const output = transpiler.transformSync(`
|
|
import deadFS from 'fs';
|
|
import anotherDeadFS from 'fs';
|
|
import liveFS from 'fs';
|
|
|
|
export var localVarToRemove = deadFS.readFileSync("/etc/passwd");
|
|
export var localVarToReplace = 1;
|
|
|
|
var getStaticProps = function () {
|
|
deadFS.readFileSync("/etc/passwd")
|
|
};
|
|
|
|
export {getStaticProps}
|
|
|
|
export function baz() {
|
|
liveFS.readFileSync("/etc/passwd");
|
|
require("bar");
|
|
}
|
|
`);
|
|
expect(output.includes("loader")).toBe(false);
|
|
expect(output.includes("react")).toBe(false);
|
|
expect(output.includes("deadFS")).toBe(false);
|
|
expect(output.includes("default")).toBe(false);
|
|
expect(output.includes("anotherDeadFS")).toBe(false);
|
|
expect(output.includes("liveFS")).toBe(true);
|
|
expect(output.includes("__N_SSG")).toBe(true);
|
|
expect(output.includes("localVarToReplace")).toBe(true);
|
|
expect(output.includes("localVarToRemove")).toBe(false);
|
|
});
|
|
});
|
|
|
|
const bunTranspiler = new Bun.Transpiler({
|
|
loader: "tsx",
|
|
define: {
|
|
"process.env.NODE_ENV": JSON.stringify("development"),
|
|
user_undefined: "undefined",
|
|
},
|
|
platform: "bun",
|
|
macro: {
|
|
inline: {
|
|
whatDidIPass: `${import.meta.dir}/inline.macro.js`,
|
|
promiseReturningFunction: `${import.meta.dir}/inline.macro.js`,
|
|
promiseReturningCtx: `${import.meta.dir}/inline.macro.js`,
|
|
},
|
|
react: {
|
|
bacon: `${import.meta.dir}/macro-check.js`,
|
|
},
|
|
},
|
|
minify: {
|
|
syntax: true,
|
|
},
|
|
});
|
|
|
|
const code = `import { useParams } from "remix";
|
|
import type { LoaderFunction, ActionFunction } from "remix";
|
|
import { type xx } from 'mod';
|
|
import { type xx as yy } from 'mod';
|
|
import { type 'xx' as yy } from 'mod';
|
|
import { type if as yy } from 'mod';
|
|
import React, { type ReactNode, Component as Romponent, Component } from 'react';
|
|
|
|
|
|
export const loader: LoaderFunction = async ({
|
|
params
|
|
}) => {
|
|
console.log(params.postId);
|
|
};
|
|
|
|
export const action: ActionFunction = async ({
|
|
params
|
|
}) => {
|
|
console.log(params.postId);
|
|
};
|
|
|
|
export default function PostRoute() {
|
|
const params = useParams();
|
|
console.log(params.postId);
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
`;
|
|
|
|
it.todo("jsxFactory (two level)", () => {
|
|
var bun = new Bun.Transpiler({
|
|
loader: "jsx",
|
|
allowBunRuntime: false,
|
|
tsconfig: JSON.stringify({
|
|
compilerOptions: {
|
|
jsxFragmentFactory: "foo.frag",
|
|
jsx: "react",
|
|
jsxFactory: "foo.factory",
|
|
},
|
|
}),
|
|
});
|
|
|
|
const element = bun.transformSync(`
|
|
export default <div>hi</div>
|
|
`);
|
|
|
|
expect(element.includes("var $jsxEl = foo.factory;")).toBe(true);
|
|
|
|
const fragment = bun.transformSync(`
|
|
export default <>hi</>
|
|
`);
|
|
expect(fragment.includes("var $JSXFrag = foo.frag,")).toBe(true);
|
|
});
|
|
|
|
it.todo("jsxFactory (one level)", () => {
|
|
var bun = new Bun.Transpiler({
|
|
loader: "jsx",
|
|
allowBunRuntime: false,
|
|
tsconfig: JSON.stringify({
|
|
compilerOptions: {
|
|
jsxFragmentFactory: "foo.frag",
|
|
jsx: "react",
|
|
jsxFactory: "h",
|
|
},
|
|
}),
|
|
});
|
|
|
|
const element = bun.transformSync(`
|
|
export default <div>hi</div>
|
|
`);
|
|
expect(element.includes("var jsxEl = h;")).toBe(true);
|
|
|
|
const fragment = bun.transformSync(`
|
|
export default <>hi</>
|
|
`);
|
|
expect(fragment.includes("var JSXFrag = foo.frag,")).toBe(true);
|
|
});
|
|
|
|
it('logLevel: "error" throws', () => {
|
|
var bun = new Bun.Transpiler({
|
|
loader: "jsx",
|
|
define: {
|
|
"process.env.NODE_ENV": JSON.stringify("development"),
|
|
},
|
|
logLevel: "error",
|
|
});
|
|
|
|
expect(() => bun.transformSync("bad??!?!?!")).toThrow("Unexpected ?");
|
|
});
|
|
|
|
it("invalid logLevel throws", () => {
|
|
expect(
|
|
() =>
|
|
new Bun.Transpiler({
|
|
loader: "jsx",
|
|
define: {
|
|
"process.env.NODE_ENV": JSON.stringify("development"),
|
|
},
|
|
logLevel: "poop",
|
|
}),
|
|
).toThrow();
|
|
});
|
|
|
|
it("JSX keys", () => {
|
|
var bun = new Bun.Transpiler({
|
|
loader: "jsx",
|
|
define: {
|
|
"process.env.NODE_ENV": JSON.stringify("development"),
|
|
},
|
|
logLevel: "error",
|
|
});
|
|
|
|
expect(bun.transformSync("console.log(<div key={() => {}} points={() => {}}></div>);")).toBe(
|
|
`console.log(jsxDEV("div", {
|
|
points: () => {
|
|
}
|
|
}, () => {
|
|
}, false, undefined, this));
|
|
`,
|
|
);
|
|
|
|
expect(bun.transformSync("console.log(<div points={() => {}} key={() => {}}></div>);")).toBe(
|
|
`console.log(jsxDEV("div", {
|
|
points: () => {
|
|
}
|
|
}, () => {
|
|
}, false, undefined, this));
|
|
`,
|
|
);
|
|
|
|
expect(bun.transformSync("console.log(<div key={() => {}} key={() => {}}></div>);")).toBe(
|
|
'console.log(jsxDEV("div", {\n key: () => {\n }\n}, () => {\n}, false, undefined, this));\n',
|
|
);
|
|
|
|
expect(bun.transformSync("console.log(<div key={() => {}}></div>, () => {});")).toBe(
|
|
'console.log(jsxDEV("div", {}, () => {\n}, false, undefined, this), () => {\n});\n',
|
|
);
|
|
|
|
expect(bun.transformSync("console.log(<div key={() => {}} a={() => {}} key={() => {}}></div>, () => {});")).toBe(
|
|
'console.log(jsxDEV("div", {\n key: () => {\n },\n a: () => {\n }\n}, () => {\n}, false, undefined, this), () => {\n});\n',
|
|
);
|
|
|
|
expect(bun.transformSync("console.log(<div key={() => {}} key={() => {}} a={() => {}}></div>, () => {});")).toBe(
|
|
'console.log(jsxDEV("div", {\n key: () => {\n },\n a: () => {\n }\n}, () => {\n}, false, undefined, this), () => {\n});\n',
|
|
);
|
|
|
|
expect(bun.transformSync("console.log(<div points={() => {}} key={() => {}}></div>);")).toBe(
|
|
`console.log(jsxDEV("div", {
|
|
points: () => {
|
|
}
|
|
}, () => {
|
|
}, false, undefined, this));
|
|
`,
|
|
);
|
|
|
|
expect(bun.transformSync("console.log(<div key={() => {}}></div>);")).toBe(
|
|
`console.log(jsxDEV("div", {}, () => {
|
|
}, false, undefined, this));
|
|
`,
|
|
);
|
|
|
|
expect(bun.transformSync("console.log(<div></div>);")).toBe(
|
|
`console.log(jsxDEV("div", {}, undefined, false, undefined, this));
|
|
`,
|
|
);
|
|
|
|
// key after spread props
|
|
// https://github.com/oven-sh/bun/issues/7328
|
|
expect(bun.transformSync(`console.log(<div {...obj} key="after" />, <div key="before" {...obj} />);`)).toBe(
|
|
`console.log(createElement(\"div\", {\n ...obj,\n key: \"after\"\n}), jsxDEV(\"div\", {\n ...obj\n}, \"before\", false, undefined, this));
|
|
`,
|
|
);
|
|
expect(bun.transformSync(`console.log(<div {...obj} key="after" {...obj2} />);`)).toBe(
|
|
`console.log(createElement(\"div\", {\n ...obj,\n key: \"after\",\n ...obj2\n}));
|
|
`,
|
|
);
|
|
expect(
|
|
bun.transformSync(`// @jsx foo;
|
|
console.log(<div {...obj} key="after" />);`),
|
|
).toBe(
|
|
`console.log(createElement(\"div\", {\n ...obj,\n key: \"after\"\n}));
|
|
`,
|
|
);
|
|
});
|
|
|
|
it.todo("JSX", () => {
|
|
var bun = new Bun.Transpiler({
|
|
loader: "jsx",
|
|
define: {
|
|
"process.env.NODE_ENV": JSON.stringify("development"),
|
|
},
|
|
});
|
|
expect(bun.transformSync("export var foo = <div foo />")).toBe(
|
|
`export var foo = jsxDEV("div", {
|
|
foo: true
|
|
}, undefined, false, undefined, this);
|
|
`,
|
|
);
|
|
expect(bun.transformSync("export var foo = <div foo={foo} />")).toBe(
|
|
`export var foo = jsxDEV("div", {
|
|
foo
|
|
}, undefined, false, undefined, this);
|
|
`,
|
|
);
|
|
expect(bun.transformSync("export var foo = <div {...foo} />")).toBe(
|
|
`export var foo = jsxDEV("div", {
|
|
...foo
|
|
}, undefined, false, undefined, this);
|
|
`,
|
|
);
|
|
|
|
expect(bun.transformSync("export var hi = <div {foo} />")).toBe(
|
|
`export var hi = jsxDEV("div", {
|
|
foo
|
|
}, undefined, false, undefined, this);
|
|
`,
|
|
);
|
|
expect(bun.transformSync("export var hi = <div {foo.bar.baz} />")).toBe(
|
|
`export var hi = jsxDEV("div", {
|
|
baz: foo.bar.baz
|
|
}, undefined, false, undefined, this);
|
|
`,
|
|
);
|
|
expect(bun.transformSync("export var hi = <div {foo?.bar?.baz} />")).toBe(
|
|
`export var hi = jsxDEV("div", {
|
|
baz: foo?.bar?.baz
|
|
}, undefined, false, undefined, this);
|
|
`,
|
|
);
|
|
expect(bun.transformSync("export var hi = <div {foo['baz'].bar?.baz} />")).toBe(
|
|
`export var hi = jsxDEV("div", {
|
|
baz: foo["baz"].bar?.baz
|
|
}, undefined, false, undefined, this);
|
|
`,
|
|
);
|
|
|
|
// cursed
|
|
expect(bun.transformSync("export var hi = <div {foo[() => true].hi} />")).toBe(
|
|
`export var hi = jsxDEV("div", {
|
|
hi: foo[() => true].hi
|
|
}, undefined, false, undefined, this);
|
|
`,
|
|
);
|
|
expect(bun.transformSync("export var hi = <Foo {process.env.NODE_ENV} />")).toBe(
|
|
`export var hi = jsxDEV(Foo, {
|
|
NODE_ENV: "development"
|
|
}, undefined, false, undefined, this);
|
|
`,
|
|
);
|
|
|
|
expect(bun.transformSync("export var hi = <div {foo['baz'].bar?.baz} />")).toBe(
|
|
`export var hi = jsxDEV("div", {
|
|
baz: foo["baz"].bar?.baz
|
|
}, undefined, false, undefined, this);
|
|
`,
|
|
);
|
|
try {
|
|
bun.transformSync("export var hi = <div {foo}={foo}= />");
|
|
throw new Error("Expected error");
|
|
} catch (e) {
|
|
expect(e.errors[0].message.includes('Expected ">"')).toBe(true);
|
|
}
|
|
|
|
expect(bun.transformSync("export var hi = <div {Foo}><Foo></Foo></div>")).toBe(
|
|
`export var hi = jsxDEV("div", {
|
|
Foo,
|
|
children: jsxDEV(Foo, {}, undefined, false, undefined, this)
|
|
}, undefined, false, undefined, this);
|
|
`,
|
|
);
|
|
expect(bun.transformSync("export var hi = <div {Foo}><Foo></Foo></div>")).toBe(
|
|
`export var hi = jsxDEV("div", {
|
|
Foo,
|
|
children: jsxDEV(Foo, {}, undefined, false, undefined, this)
|
|
}, undefined, false, undefined, this);
|
|
`,
|
|
);
|
|
|
|
expect(bun.transformSync("export var hi = <div>{123}}</div>").trim()).toBe(
|
|
`export var hi = jsxDEV("div", {
|
|
children: [
|
|
123,
|
|
"}"
|
|
]
|
|
}, undefined, true, undefined, this);
|
|
`.trim(),
|
|
);
|
|
});
|
|
|
|
describe("inline JSX", () => {
|
|
const inliner = new Bun.Transpiler({
|
|
loader: "tsx",
|
|
define: {
|
|
"process.env.NODE_ENV": JSON.stringify("production"),
|
|
user_undefined: "undefined",
|
|
},
|
|
platform: "bun",
|
|
jsxOptimizationInline: true,
|
|
treeShaking: false,
|
|
inline: true,
|
|
deadCodeElimination: true,
|
|
allowBunRuntime: true,
|
|
|
|
target: "bun",
|
|
tsconfig: JSON.stringify({
|
|
compilerOptions: {
|
|
jsxImportSource: "react",
|
|
},
|
|
}),
|
|
});
|
|
|
|
it("inlines static JSX into object literals", () => {
|
|
expect(
|
|
inliner
|
|
.transformSync(
|
|
`
|
|
export var hi = <div>{123}</div>
|
|
export var hiWithKey = <div key="hey">{123}</div>
|
|
export var hiWithRef = <div ref={foo}>{123}</div>
|
|
|
|
export var ComponentThatChecksDefaultProps = <Hello></Hello>
|
|
export var ComponentThatChecksDefaultPropsAndHasChildren = <Hello>my child</Hello>
|
|
export var ComponentThatHasSpreadCausesDeopt = <Hello {...spread} />
|
|
|
|
`.trim(),
|
|
)
|
|
.replaceAll("\n", "")
|
|
.replaceAll(" ", "")
|
|
.trim(),
|
|
).toBe(
|
|
// TODO: figure out why its using jsxDEV() here. It doesn't do that with NODE_ENV=production at runtime.
|
|
`
|
|
import {
|
|
$$typeof as $$typeof_4ad651bb3f5de058,
|
|
__merge as __merge_e79ebbbc0cc1f55b
|
|
} from "bun:wrap";
|
|
export var hi = {
|
|
$$typeof: $$typeof_4ad651bb3f5de058,
|
|
type: "div",
|
|
key: null,
|
|
ref: null,
|
|
props: {
|
|
children: 123
|
|
},
|
|
_owner: null
|
|
}, hiWithKey = {
|
|
$$typeof: $$typeof_4ad651bb3f5de058,
|
|
type: "div",
|
|
key: "hey",
|
|
ref: null,
|
|
props: {
|
|
children: 123
|
|
},
|
|
_owner: null
|
|
}, hiWithRef = jsxDEV("div", {
|
|
ref: foo,
|
|
children: 123
|
|
}, void 0, !1, void 0, this), ComponentThatChecksDefaultProps = {
|
|
$$typeof: $$typeof_4ad651bb3f5de058,
|
|
type: Hello,
|
|
key: null,
|
|
ref: null,
|
|
props: Hello.defaultProps || {},
|
|
_owner: null
|
|
}, ComponentThatChecksDefaultPropsAndHasChildren = {
|
|
$$typeof: $$typeof_4ad651bb3f5de058,
|
|
type: Hello,
|
|
key: null,
|
|
ref: null,
|
|
props: __merge_e79ebbbc0cc1f55b({
|
|
children: "my child"
|
|
}, Hello.defaultProps),
|
|
_owner: null
|
|
}, ComponentThatHasSpreadCausesDeopt = jsxDEV(Hello, {
|
|
...spread
|
|
}, void 0, !1, void 0, this);
|
|
`
|
|
.replaceAll("\n", "")
|
|
.replaceAll(" ", "")
|
|
.trim(),
|
|
);
|
|
});
|
|
});
|
|
|
|
it("JSX spread children", () => {
|
|
var bun = new Bun.Transpiler({
|
|
loader: "jsx",
|
|
define: {
|
|
"process.env.NODE_ENV": JSON.stringify("development"),
|
|
},
|
|
});
|
|
expect(bun.transformSync("export var foo = <div>{...a}b</div>")).toBe(
|
|
`export var foo = jsxDEV("div", {
|
|
children: [
|
|
...a,
|
|
"b"
|
|
]
|
|
}, undefined, true, undefined, this);
|
|
`,
|
|
);
|
|
|
|
expect(bun.transformSync("export var foo = <div>{...a}</div>")).toBe(
|
|
`export var foo = jsxDEV("div", {
|
|
children: [...a]
|
|
}, undefined, true, undefined, this);
|
|
`,
|
|
);
|
|
});
|
|
|
|
it("require with a dynamic non-string expression", () => {
|
|
var nodeTranspiler = new Bun.Transpiler({ platform: "node" });
|
|
expect(nodeTranspiler.transformSync("require('hi' + bar)")).toBe('require("hi" + bar);\n');
|
|
});
|
|
|
|
it("CommonJS", () => {
|
|
var nodeTranspiler = new Bun.Transpiler({ platform: "node" });
|
|
expect(nodeTranspiler.transformSync("module.require('hi' + 123)")).toBe('require("hi" + 123);\n');
|
|
|
|
expect(nodeTranspiler.transformSync("module.require(1 ? 'foo' : 'bar')")).toBe('require("foo");\n');
|
|
expect(nodeTranspiler.transformSync("require(1 ? 'foo' : 'bar')")).toBe('require("foo");\n');
|
|
|
|
expect(nodeTranspiler.transformSync("module.require(unknown ? 'foo' : 'bar')")).toBe(
|
|
'unknown ? require("foo") : require("bar");\n',
|
|
);
|
|
});
|
|
|
|
describe("regressions", () => {
|
|
it("unexpected super", () => {
|
|
const input = `
|
|
'use strict';
|
|
|
|
const ErrorReportingMixinBase = require('./mixin-base');
|
|
const PositionTrackingPreprocessorMixin = require('../position-tracking/preprocessor-mixin');
|
|
const Mixin = require('../../utils/mixin');
|
|
|
|
class ErrorReportingPreprocessorMixin extends ErrorReportingMixinBase {
|
|
constructor(preprocessor, opts) {
|
|
super(preprocessor, opts);
|
|
|
|
this.posTracker = Mixin.install(preprocessor, PositionTrackingPreprocessorMixin);
|
|
this.lastErrOffset = -1;
|
|
}
|
|
|
|
_reportError(code) {
|
|
//NOTE: avoid reporting error twice on advance/retreat
|
|
if (this.lastErrOffset !== this.posTracker.offset) {
|
|
this.lastErrOffset = this.posTracker.offset;
|
|
super._reportError(code);
|
|
}
|
|
}
|
|
}
|
|
|
|
module.exports = ErrorReportingPreprocessorMixin;
|
|
|
|
|
|
`;
|
|
expect(transpiler.transformSync(input, "js").length > 0).toBe(true);
|
|
});
|
|
});
|
|
|
|
describe("scanImports", () => {
|
|
it("reports import paths, excluding types", () => {
|
|
const imports = transpiler.scanImports(code, "tsx");
|
|
expect(imports.filter(({ path }) => path === "remix")).toHaveLength(1);
|
|
expect(imports.filter(({ path }) => path === "mod")).toHaveLength(0);
|
|
expect(imports.filter(({ path }) => path === "react")).toHaveLength(1);
|
|
expect(imports).toHaveLength(2);
|
|
});
|
|
});
|
|
|
|
const parsed = (code, trim = true, autoExport = false, transpiler_ = transpiler) => {
|
|
if (autoExport) {
|
|
code = "export default (" + code + ")";
|
|
}
|
|
|
|
var out = transpiler_.transformSync(code, "js");
|
|
if (autoExport && out.startsWith("export default ")) {
|
|
out = out.substring("export default ".length);
|
|
}
|
|
|
|
if (trim) {
|
|
out = out.trim();
|
|
|
|
if (out.endsWith(";")) {
|
|
out = out.substring(0, out.length - 1);
|
|
}
|
|
|
|
return out.trim();
|
|
}
|
|
|
|
return out;
|
|
};
|
|
|
|
const expectPrinted = (code, out) => {
|
|
expect(parsed(code, true, true)).toBe(out);
|
|
};
|
|
|
|
const expectPrinted_ = (code, out) => {
|
|
expect(parsed(code, !out.endsWith(";\n"), false)).toBe(out);
|
|
};
|
|
|
|
const expectPrintedNoTrim = (code, out) => {
|
|
expect(parsed(code, false, false)).toBe(out);
|
|
};
|
|
|
|
const expectBunPrinted_ = (code, out) => {
|
|
expect(parsed(code, !out.endsWith(";\n"), false, bunTranspiler)).toBe(out);
|
|
};
|
|
|
|
const expectParseError = (code, message) => {
|
|
try {
|
|
parsed(code, false, false);
|
|
} catch (er) {
|
|
var err = er;
|
|
if (er instanceof AggregateError) {
|
|
err = err.errors[0];
|
|
}
|
|
|
|
expect(er.message).toBe(message);
|
|
|
|
return;
|
|
}
|
|
|
|
throw new Error("Expected parse error for code\n\t" + code);
|
|
};
|
|
|
|
describe("parser", () => {
|
|
it("arrays", () => {
|
|
expectPrinted("[]", "[]");
|
|
expectPrinted("[,]", "[,]");
|
|
expectPrinted("[1]", "[1]");
|
|
expectPrinted("[1,]", "[1]");
|
|
expectPrinted("[,1]", "[, 1]");
|
|
expectPrinted("[1,2]", "[1, 2]");
|
|
expectPrinted("[,1,2]", "[, 1, 2]");
|
|
expectPrinted("[1,,2]", "[1, , 2]");
|
|
expectPrinted("[1,2,]", "[1, 2]");
|
|
expectPrinted("[1,2,,]", "[1, 2, ,]");
|
|
});
|
|
|
|
it("exponentiation", () => {
|
|
expectPrinted("(delete x) ** 0", "(delete x) ** 0");
|
|
expectPrinted("(delete x.prop) ** 0", "(delete x.prop) ** 0");
|
|
expectPrinted("(delete x[0]) ** 0", "(delete x[0]) ** 0");
|
|
|
|
expectPrinted("(delete x?.prop) ** 0", "(delete x?.prop) ** 0");
|
|
|
|
expectPrinted("(void x) ** 0", "(void x) ** 0");
|
|
expectPrinted("(typeof x) ** 0", "(typeof x) ** 0");
|
|
expectPrinted("(+x) ** 0", "(+x) ** 0");
|
|
expectPrinted("(-x) ** 0", "(-x) ** 0");
|
|
expectPrinted("(~x) ** 0", "(~x) ** 0");
|
|
expectPrinted("(!x) ** 0", "(!x) ** 0");
|
|
expectPrinted("(await x) ** 0", "(await x) ** 0");
|
|
expectPrinted("(await -x) ** 0", "(await -x) ** 0");
|
|
|
|
expectPrinted("--x ** 2", "--x ** 2");
|
|
expectPrinted("++x ** 2", "++x ** 2");
|
|
expectPrinted("x-- ** 2", "x-- ** 2");
|
|
expectPrinted("x++ ** 2", "x++ ** 2");
|
|
|
|
expectPrinted("(-x) ** 2", "(-x) ** 2");
|
|
expectPrinted("(+x) ** 2", "(+x) ** 2");
|
|
expectPrinted("(~x) ** 2", "(~x) ** 2");
|
|
expectPrinted("(!x) ** 2", "(!x) ** 2");
|
|
expectPrinted("(-1) ** 2", "(-1) ** 2");
|
|
expectPrinted("(+1) ** 2", "1 ** 2");
|
|
expectPrinted("(~1) ** 2", "(~1) ** 2");
|
|
expectPrinted("(!1) ** 2", "false ** 2");
|
|
expectPrinted("(void x) ** 2", "(void x) ** 2");
|
|
expectPrinted("(delete x) ** 2", "(delete x) ** 2");
|
|
expectPrinted("(typeof x) ** 2", "(typeof x) ** 2");
|
|
expectPrinted("undefined ** 2", "undefined ** 2");
|
|
|
|
expectParseError("-x ** 2", "Unexpected **");
|
|
expectParseError("+x ** 2", "Unexpected **");
|
|
expectParseError("~x ** 2", "Unexpected **");
|
|
expectParseError("!x ** 2", "Unexpected **");
|
|
expectParseError("void x ** 2", "Unexpected **");
|
|
expectParseError("delete x ** 2", "Unexpected **");
|
|
expectParseError("typeof x ** 2", "Unexpected **");
|
|
|
|
expectParseError("-x.y() ** 2", "Unexpected **");
|
|
expectParseError("+x.y() ** 2", "Unexpected **");
|
|
expectParseError("~x.y() ** 2", "Unexpected **");
|
|
expectParseError("!x.y() ** 2", "Unexpected **");
|
|
expectParseError("void x.y() ** 2", "Unexpected **");
|
|
expectParseError("delete x.y() ** 2", "Unexpected **");
|
|
expectParseError("typeof x.y() ** 2", "Unexpected **");
|
|
|
|
expectParseError("delete x ** 0", "Unexpected **");
|
|
expectParseError("delete x.prop ** 0", "Unexpected **");
|
|
expectParseError("delete x[0] ** 0", "Unexpected **");
|
|
expectParseError("delete x?.prop ** 0", "Unexpected **");
|
|
expectParseError("void x ** 0", "Unexpected **");
|
|
expectParseError("typeof x ** 0", "Unexpected **");
|
|
expectParseError("+x ** 0", "Unexpected **");
|
|
expectParseError("-x ** 0", "Unexpected **");
|
|
expectParseError("~x ** 0", "Unexpected **");
|
|
expectParseError("!x ** 0", "Unexpected **");
|
|
expectParseError("await x ** 0", "Unexpected **");
|
|
expectParseError("await -x ** 0", "Unexpected **");
|
|
});
|
|
|
|
it("await", () => {
|
|
expectPrinted("await x", "await x");
|
|
expectPrinted("await +x", "await +x");
|
|
expectPrinted("await -x", "await -x");
|
|
expectPrinted("await ~x", "await ~x");
|
|
expectPrinted("await !x", "await !x");
|
|
expectPrinted("await --x", "await --x");
|
|
expectPrinted("await ++x", "await ++x");
|
|
expectPrinted("await x--", "await x--");
|
|
expectPrinted("await x++", "await x++");
|
|
expectPrinted("await void x", "await void x");
|
|
expectPrinted("await typeof x", "await typeof x");
|
|
expectPrinted("await (x * y)", "await (x * y)");
|
|
expectPrinted("await (x ** y)", "await (x ** y)");
|
|
|
|
expectPrinted_("async function f() { await delete x }", "async function f() {\n await delete x;\n}");
|
|
|
|
// expectParseError(
|
|
// "await delete x",
|
|
// "Delete of a bare identifier cannot be used in an ECMAScript module"
|
|
// );
|
|
});
|
|
|
|
it("import assert", () => {
|
|
expectPrinted_(`import json from "./foo.json" assert { type: "json" };`, `import json from "./foo.json"`);
|
|
expectPrinted_(`import json from "./foo.json";`, `import json from "./foo.json"`);
|
|
expectPrinted_(`import("./foo.json", { type: "json" });`, `import("./foo.json")`);
|
|
});
|
|
|
|
it("import with unicode escape", () => {
|
|
expectPrinted_(`import { name } from 'mod\\u1011';`, `import {name} from "mod\\u1011"`);
|
|
});
|
|
|
|
it("fold string addition", () => {
|
|
expectPrinted_(
|
|
`
|
|
const a = "[^aeiou]";
|
|
const b = a + "[^aeiouy]*";
|
|
console.log(a);
|
|
`,
|
|
`
|
|
const a = "[^aeiou]";
|
|
const b = a + "[^aeiouy]*";
|
|
console.log(a)
|
|
`.trim(),
|
|
);
|
|
|
|
expectPrinted_(`export const foo = "a" + "b";`, `export const foo = "ab"`);
|
|
expectPrinted_(
|
|
`export const foo = "F" + "0" + "F" + "0123456789" + "ABCDEF" + "0123456789ABCDEFF0123456789ABCDEF00" + "b";`,
|
|
`export const foo = "F0F0123456789ABCDEF0123456789ABCDEFF0123456789ABCDEF00b"`,
|
|
);
|
|
expectPrinted_(`export const foo = "a" + 1 + "b";`, `export const foo = "a" + 1 + "b"`);
|
|
expectPrinted_(`export const foo = "a" + "b" + 1 + "b";`, `export const foo = "ab" + 1 + "b"`);
|
|
expectPrinted_(`export const foo = "a" + "b" + 1 + "b" + "c";`, `export const foo = "ab" + 1 + "bc"`);
|
|
});
|
|
|
|
it("numeric constants", () => {
|
|
expectBunPrinted_("export const foo = 1 + 2", "export const foo = 3");
|
|
expectBunPrinted_("export const foo = 1 - 2", "export const foo = -1");
|
|
expectBunPrinted_("export const foo = 1 * 2", "export const foo = 2");
|
|
});
|
|
|
|
it.todo("pass objects to macros", () => {
|
|
var object = {
|
|
helloooooooo: {
|
|
message: [12345],
|
|
},
|
|
};
|
|
|
|
const output = bunTranspiler.transformSync(
|
|
`
|
|
import {whatDidIPass} from 'inline';
|
|
|
|
export function foo() {
|
|
return whatDidIPass();
|
|
}
|
|
`,
|
|
object,
|
|
);
|
|
expect(output).toBe(`export function foo() {
|
|
return {
|
|
helloooooooo: {
|
|
message: [
|
|
12345
|
|
]
|
|
}
|
|
};
|
|
}
|
|
`);
|
|
});
|
|
|
|
it.todo("macros can return a promise", () => {
|
|
var object = {
|
|
helloooooooo: {
|
|
message: [12345],
|
|
},
|
|
};
|
|
|
|
const output = bunTranspiler.transformSync(
|
|
`
|
|
import {promiseReturningFunction} from 'inline';
|
|
|
|
export function foo() {
|
|
return promiseReturningFunction();
|
|
}
|
|
`,
|
|
object,
|
|
);
|
|
expect(output).toBe(`export function foo() {
|
|
return 1;
|
|
}
|
|
`);
|
|
});
|
|
|
|
it.todo("macros can return a Response body", () => {
|
|
// "promiseReturningCtx" is this:
|
|
// export function promiseReturningCtx(expr, ctx) {
|
|
// return new Promise((resolve, reject) => {
|
|
// setTimeout(() => {
|
|
// resolve(ctx);
|
|
// }, 1);
|
|
// });
|
|
// }
|
|
var object = Response.json({ hello: "world" });
|
|
|
|
const input = `
|
|
import {promiseReturningCtx} from 'inline';
|
|
|
|
export function foo() {
|
|
return promiseReturningCtx();
|
|
}
|
|
`.trim();
|
|
|
|
const output = `
|
|
export function foo() {
|
|
return { hello: "world" };
|
|
}
|
|
`.trim();
|
|
|
|
expect(bunTranspiler.transformSync(input, object).trim()).toBe(output);
|
|
});
|
|
|
|
it.todo("macros get dead code eliminated", () => {
|
|
var object = Response.json({
|
|
big: {
|
|
object: {
|
|
beep: "boop",
|
|
huge: 123,
|
|
},
|
|
blobby: {
|
|
beep: "boop",
|
|
huge: 123,
|
|
},
|
|
},
|
|
dead: "hello world!",
|
|
});
|
|
|
|
const input = `
|
|
import {promiseReturningCtx} from 'inline';
|
|
|
|
export const {dead} = promiseReturningCtx();
|
|
`.trim();
|
|
|
|
const output = `
|
|
export const { dead } = { dead: "hello world!" };
|
|
`.trim();
|
|
|
|
expect(bunTranspiler.transformSync(input, object).trim()).toBe(output);
|
|
});
|
|
|
|
it.skip("rewrite string to length", () => {
|
|
expectBunPrinted_(`export const foo = "a".length + "b".length;`, `export const foo = 2`);
|
|
// check rope string
|
|
expectBunPrinted_(`export const foo = ("a" + "b").length;`, `export const foo = 2`);
|
|
expectBunPrinted_(
|
|
// check UTF-16
|
|
`export const foo = "😋 Get Emoji — All Emojis to ✂️ Copy and 📋 Paste 👌".length;`,
|
|
`export const foo = 52`,
|
|
);
|
|
});
|
|
|
|
describe("Bun.js", () => {
|
|
it.todo("require -> import.meta.require", () => {
|
|
expectBunPrinted_(
|
|
`export const foo = require('bar.node')`,
|
|
`export const foo = import.meta.require("bar.node")`,
|
|
);
|
|
expectBunPrinted_(
|
|
`export const foo = require('bar.node')`,
|
|
`export const foo = import.meta.require("bar.node")`,
|
|
);
|
|
});
|
|
|
|
it.todo("require.resolve -> import.meta.require.resolve", () => {
|
|
expectBunPrinted_(
|
|
`export const foo = require.resolve('bar.node')`,
|
|
`export const foo = import.meta.require.resolve("bar.node")`,
|
|
);
|
|
});
|
|
|
|
it.todo('require.resolve(path, {paths: ["blah"]}) -> import.meta.require.resolve', () => {
|
|
expectBunPrinted_(
|
|
`export const foo = require.resolve('bar.node', {paths: ["blah"]})`,
|
|
`export const foo = import.meta.require.resolve("bar.node", { paths: ["blah"] })`,
|
|
);
|
|
});
|
|
|
|
it.todo("require is defined", () => {
|
|
expectBunPrinted_(
|
|
`
|
|
const {resolve} = require;
|
|
console.log(resolve.length)
|
|
`.trim(),
|
|
`
|
|
const { resolve } = import.meta.require;
|
|
console.log(resolve.length)
|
|
`.trim(),
|
|
);
|
|
});
|
|
});
|
|
|
|
describe("Browsers", () => {
|
|
it('require.resolve("my-module") is untouched', () => {
|
|
// we used to inline the string for this, but that is always incorrect as require.resolve builds an exact path.
|
|
expectPrinted_(
|
|
`export const foo = require.resolve('my-module')`,
|
|
`export const foo = require.resolve("my-module")`,
|
|
);
|
|
});
|
|
});
|
|
|
|
it("define", () => {
|
|
expectPrinted_(`export default typeof user_undefined === 'undefined';`, `export default true`);
|
|
expectPrinted_(`export default typeof user_undefined !== 'undefined';`, `export default false`);
|
|
|
|
expectPrinted_(`export default typeof user_undefined !== 'undefined';`, `export default false`);
|
|
expectPrinted_(`export default !user_undefined;`, `export default true`);
|
|
|
|
expectPrinted_(`export default user_nested;`, `export default location.origin`);
|
|
expectPrinted_("hello.earth('hi')", 'hello.mars("hi")');
|
|
expectPrinted_("Math.log('hi')", 'console.error("hi")');
|
|
});
|
|
|
|
it("jsx symbol should work", () => {
|
|
expectBunPrinted_(`var x = jsx; export default x;`, "var x = jsx;\nexport default x");
|
|
});
|
|
|
|
it.todo("decls", () => {
|
|
// expectParseError("var x = 0", "");
|
|
// expectParseError("let x = 0", "");
|
|
// expectParseError("const x = 0", "");
|
|
// expectParseError("for (var x = 0;;) ;", "");
|
|
// expectParseError("for (let x = 0;;) ;", "");
|
|
// expectParseError("for (const x = 0;;) ;", "");
|
|
|
|
// expectParseError("for (var x in y) ;", "");
|
|
// expectParseError("for (let x in y) ;", "");
|
|
// expectParseError("for (const x in y) ;", "");
|
|
// expectParseError("for (var x of y) ;", "");
|
|
// expectParseError("for (let x of y) ;", "");
|
|
// expectParseError("for (const x of y) ;", "");
|
|
|
|
// expectParseError("var x", "");
|
|
// expectParseError("let x", "");
|
|
expectParseError("const x", 'The constant "x" must be initialized');
|
|
expectParseError("const {}", "This constant must be initialized");
|
|
expectParseError("const []", "This constant must be initialized");
|
|
// expectParseError("for (var x;;) ;", "");
|
|
// expectParseError("for (let x;;) ;", "");
|
|
expectParseError("for (const x;;) ;", 'The constant "x" must be initialized');
|
|
expectParseError("for (const {};;) ;", "This constant must be initialized");
|
|
expectParseError("for (const [];;) ;", "This constant must be initialized");
|
|
|
|
// Make sure bindings are visited during parsing
|
|
expectPrinted_("var {[x]: y} = {}", "var { [x]: y } = {}");
|
|
expectPrinted_("var {...x} = {}", "var { ...x } = {}");
|
|
|
|
// Test destructuring patterns
|
|
expectPrinted_("var [...x] = []", "var [...x] = []");
|
|
expectPrinted_("var {...x} = {}", "var { ...x } = {}");
|
|
|
|
expectPrinted_("export var foo = ([...x] = []) => {}", "export var foo = ([...x] = []) => {\n}");
|
|
|
|
expectPrinted_("export var foo = ({...x} = {}) => {}", "export var foo = ({ ...x } = {}) => {\n}");
|
|
|
|
expectParseError("var [...x,] = []", 'Unexpected "," after rest pattern');
|
|
expectParseError("var {...x,} = {}", 'Unexpected "," after rest pattern');
|
|
expectParseError(
|
|
"export default function() { return ([...x,] = []) => {} }",
|
|
"Unexpected trailing comma after rest element",
|
|
);
|
|
expectParseError("({...x,} = {}) => {}", "Unexpected trailing comma after rest element");
|
|
|
|
expectPrinted_("[b, ...c] = d", "[b, ...c] = d");
|
|
expectPrinted_("([b, ...c] = d)", "[b, ...c] = d");
|
|
expectPrinted_("({b, ...c} = d)", "({ b, ...c } = d)");
|
|
expectPrinted_("({a = b} = c)", "({ a = b } = c)");
|
|
expectPrinted_("({a: b = c} = d)", "({ a: b = c } = d)");
|
|
expectPrinted_("({a: b.c} = d)", "({ a: b.c } = d)");
|
|
expectPrinted_("[a = {}] = b", "[a = {}] = b");
|
|
expectPrinted_("[[...a, b].x] = c", "[[...a, b].x] = c");
|
|
expectPrinted_("[{...a, b}.x] = c", "[{ ...a, b }.x] = c");
|
|
expectPrinted_("({x: [...a, b].x} = c)", "({ x: [...a, b].x } = c)");
|
|
expectPrinted_("({x: {...a, b}.x} = c)", "({ x: { ...a, b }.x } = c)");
|
|
expectPrinted_("[x = [...a, b]] = c", "[x = [...a, b]] = c");
|
|
expectPrinted_("[x = {...a, b}] = c", "[x = { ...a, b }] = c");
|
|
expectPrinted_("({x = [...a, b]} = c)", "({ x = [...a, b] } = c)");
|
|
expectPrinted_("({x = {...a, b}} = c)", "({ x = { ...a, b } } = c)");
|
|
|
|
expectPrinted_("(x = y)", "x = y");
|
|
expectPrinted_("([] = [])", "[] = []");
|
|
expectPrinted_("({} = {})", "({} = {})");
|
|
expectPrinted_("([[]] = [[]])", "[[]] = [[]]");
|
|
expectPrinted_("({x: {}} = {x: {}})", "({ x: {} } = { x: {} })");
|
|
expectPrinted_("(x) = y", "x = y");
|
|
expectParseError("([]) = []", "Invalid assignment target");
|
|
expectParseError("({}) = {}", "Invalid assignment target");
|
|
expectParseError("[([])] = [[]]", "Invalid assignment target");
|
|
expectParseError("({x: ({})} = {x: {}})", "Invalid assignment target");
|
|
expectParseError("(([]) = []) => {}", "Unexpected parentheses in binding pattern");
|
|
expectParseError("(({}) = {}) => {}", "Unexpected parentheses in binding pattern");
|
|
expectParseError("function f(([]) = []) {}", "Parse error");
|
|
expectParseError(
|
|
"function f(({}) = {}) {}",
|
|
"Parse error",
|
|
// 'Expected identifier but found "("\n'
|
|
);
|
|
|
|
expectPrintedNoTrim("for (x in y) ;", "for (x in y)\n ;\n");
|
|
expectPrintedNoTrim("for ([] in y) ;", "for ([] in y)\n ;\n");
|
|
expectPrintedNoTrim("for ({} in y) ;", "for ({} in y)\n ;\n");
|
|
expectPrintedNoTrim("for ((x) in y) ;", "for (x in y)\n ;\n");
|
|
expectParseError("for (x in y)", "Unexpected end of file");
|
|
expectParseError("for ([] in y)", "Unexpected end of file");
|
|
expectParseError("for ({} in y)", "Unexpected end of file");
|
|
expectParseError("for ((x) in y)", "Unexpected end of file");
|
|
expectParseError("for (([]) in y) ;", "Invalid assignment target");
|
|
expectParseError("for (({}) in y) ;", "Invalid assignment target");
|
|
|
|
expectPrintedNoTrim("for (x of y) ;", "for (x of y)\n ;\n");
|
|
expectPrintedNoTrim("for ([] of y) ;", "for ([] of y)\n ;\n");
|
|
expectPrintedNoTrim("for ({} of y) ;", "for ({} of y)\n ;\n");
|
|
expectPrintedNoTrim("for ((x) of y) ;", "for (x of y)\n ;\n");
|
|
expectPrintedNoTrim("for (x of y) {}", "for (x of y)\n ;\n");
|
|
expectPrintedNoTrim("for ([] of y) {}", "for ([] of y)\n ;\n");
|
|
expectPrintedNoTrim("for ({} of y) {}", "for ({} of y)\n ;\n");
|
|
expectPrintedNoTrim("for ((x) of y) {}", "for (x of y)\n ;\n");
|
|
expectParseError("for (x of y)", "Unexpected end of file");
|
|
expectParseError("for ([] of y)", "Unexpected end of file");
|
|
expectParseError("for ({} of y)", "Unexpected end of file");
|
|
expectParseError("for ((x) of y)", "Unexpected end of file");
|
|
expectParseError("for (([]) of y) ;", "Invalid assignment target");
|
|
expectParseError("for (({}) of y) ;", "Invalid assignment target");
|
|
|
|
expectParseError("[[...a, b]] = c", 'Unexpected "," after rest pattern');
|
|
expectParseError("[{...a, b}] = c", 'Unexpected "," after rest pattern');
|
|
expectParseError("({x: [...a, b]} = c)", 'Unexpected "," after rest pattern');
|
|
expectParseError("({x: {...a, b}} = c)", 'Unexpected "," after rest pattern');
|
|
expectParseError("[b, ...c,] = d", 'Unexpected "," after rest pattern');
|
|
expectParseError("([b, ...c,] = d)", 'Unexpected "," after rest pattern');
|
|
expectParseError("({b, ...c,} = d)", 'Unexpected "," after rest pattern');
|
|
expectParseError("({a = b})", 'Unexpected "="');
|
|
expectParseError("({x = {a = b}} = c)", 'Unexpected "="');
|
|
expectParseError("[a = {b = c}] = d", 'Unexpected "="');
|
|
|
|
expectPrintedNoTrim("for ([{a = {}}] in b) ;", "for ([{ a = {} }] in b)\n ;\n");
|
|
expectPrintedNoTrim("for ([{a = {}}] of b) ;", "for ([{ a = {} }] of b)\n ;\n");
|
|
expectPrintedNoTrim("for ({a = {}} in b) ;", "for ({ a = {} } in b)\n ;\n");
|
|
expectPrintedNoTrim("for ({a = {}} of b) ;", "for ({ a = {} } of b)\n ;\n");
|
|
expectParseError("for ([{a = {}}] in b)", "Unexpected end of file");
|
|
expectParseError("for ([{a = {}}] of b)", "Unexpected end of file");
|
|
expectParseError("for ({a = {}} in b)", "Unexpected end of file");
|
|
expectParseError("for ({a = {}} of b)", "Unexpected end of file");
|
|
|
|
// this is different from esbuild
|
|
expectPrintedNoTrim("for ([{a = {}}] in b) {}", "for ([{ a = {} }] in b)\n ;\n");
|
|
expectPrintedNoTrim("for ([{a = {}}] of b) {}", "for ([{ a = {} }] of b)\n ;\n");
|
|
expectPrintedNoTrim("for ({a = {}} in b) {}", "for ({ a = {} } in b)\n ;\n");
|
|
expectPrintedNoTrim("for ({a = {}} of b) {}", "for ({ a = {} } of b)\n ;\n");
|
|
expectPrintedNoTrim("for (x in y) {}", "for (x in y)\n ;\n");
|
|
expectPrintedNoTrim("for ([] in y) {}", "for ([] in y)\n ;\n");
|
|
expectPrintedNoTrim("for ({} in y) {}", "for ({} in y)\n ;\n");
|
|
expectPrintedNoTrim("for ((x) in y) {}", "for (x in y)\n ;\n");
|
|
expectPrintedNoTrim("while (true) {}", "while (true)\n ;\n");
|
|
|
|
expectPrintedNoTrim("while (true) ;", "while (true)\n ;\n");
|
|
expectParseError("while (1)", "Unexpected end of file");
|
|
|
|
expectParseError("({a = {}} in b)", 'Unexpected "="');
|
|
expectParseError("[{a = {}}]\nof()", 'Unexpected "="');
|
|
expectParseError("for ([...a, b] in c) {}", 'Unexpected "," after rest pattern');
|
|
expectParseError("for ([...a, b] of c) {}", 'Unexpected "," after rest pattern');
|
|
});
|
|
|
|
it("regexp", () => {
|
|
expectPrinted("/x/g", "/x/g");
|
|
expectPrinted("/x/i", "/x/i");
|
|
expectPrinted("/x/m", "/x/m");
|
|
expectPrinted("/x/s", "/x/s");
|
|
expectPrinted("/x/u", "/x/u");
|
|
expectPrinted("/x/y", "/x/y");
|
|
expectPrinted("/gimme/g", "/gimme/g");
|
|
expectPrinted("/gimgim/g", "/gimgim/g");
|
|
|
|
expectParseError("/x/msuygig", 'Duplicate flag "g" in regular expression');
|
|
});
|
|
|
|
it("identifier escapes", () => {
|
|
expectPrinted_("var _\u0076\u0061\u0072", "var _var");
|
|
expectParseError("var \u0076\u0061\u0072", 'Expected identifier but found "\u0076\u0061\u0072"');
|
|
expectParseError("\\u0076\\u0061\\u0072 foo", "Unexpected \\u0076\\u0061\\u0072");
|
|
|
|
expectPrinted_("foo._\u0076\u0061\u0072", "foo._var");
|
|
expectPrinted_("foo.\u0076\u0061\u0072", "foo.var");
|
|
|
|
// expectParseError("\u200Ca", 'Unexpected "\\u200c"');
|
|
// expectParseError("\u200Da", 'Unexpected "\\u200d"');
|
|
});
|
|
|
|
describe("dead code elimination", () => {
|
|
const transpilerNoDCE = new Bun.Transpiler({ deadCodeElimination: false });
|
|
it("should DCE with deadCodeElimination: true or by default", () => {
|
|
expect(parsed("123", true, false)).toBe("");
|
|
expect(parsed("[-1, 2n, null]", true, false)).toBe("");
|
|
expect(parsed("true", true, false)).toBe("");
|
|
expect(parsed("!0", true, false)).toBe("");
|
|
expect(parsed('if (!1) "dead";', true, false)).toBe("if (false)");
|
|
expect(parsed("if (!1) var x = 2;", true, false)).toBe("if (false)\n var x");
|
|
expect(parsed("if (undefined) { let y = Math.random(); }", true, false)).toBe("if (undefined) {\n}");
|
|
});
|
|
it("should not DCE with deadCodeElimination: false", () => {
|
|
expect(parsed("123", true, false, transpilerNoDCE)).toBe("123");
|
|
expect(parsed("[1, 2n, null]", true, false, transpilerNoDCE)).toBe("[1, 2n, null]");
|
|
expect(parsed("true", true, false, transpilerNoDCE)).toBe("true");
|
|
expect(parsed("!0", true, false, transpilerNoDCE)).toBe("!0");
|
|
expect(parsed('if (!1) "dead";', true, false, transpilerNoDCE)).toBe('if (!1)\n "dead"');
|
|
expect(parsed("if (!1) var x = 2;", true, false, transpilerNoDCE)).toBe("if (!1)\n var x = 2");
|
|
expect(parsed("if (undefined) { let y = Math.random(); }", true, false, transpilerNoDCE)).toBe(
|
|
"if (undefined) {\n let y = Math.random();\n}",
|
|
);
|
|
});
|
|
});
|
|
});
|
|
|
|
it("private identifiers", () => {
|
|
expectParseError("#foo", "Unexpected #foo");
|
|
expectParseError("#foo in this", "Unexpected #foo");
|
|
expectParseError("this.#foo", 'Expected identifier but found "#foo"');
|
|
expectParseError("this?.#foo", 'Expected identifier but found "#foo"');
|
|
expectParseError("({ #foo: 1 })", 'Expected identifier but found "#foo"');
|
|
expectParseError("class Foo { x = { #foo: 1 } }", 'Expected identifier but found "#foo"');
|
|
expectParseError("class Foo { x = #foo }", 'Expected "in" but found "}"');
|
|
expectParseError("class Foo { #foo; foo() { delete this.#foo } }", 'Deleting the private name "#foo" is forbidden');
|
|
expectParseError(
|
|
"class Foo { #foo; foo() { delete this?.#foo } }",
|
|
'Deleting the private name "#foo" is forbidden',
|
|
);
|
|
expectParseError("class Foo extends Bar { #foo; foo() { super.#foo } }", 'Expected identifier but found "#foo"');
|
|
expectParseError("class Foo { #foo = () => { for (#foo in this) ; } }", "Unexpected #foo");
|
|
expectParseError("class Foo { #foo = () => { for (x = #foo in this) ; } }", "Unexpected #foo");
|
|
expectPrinted_("class Foo { #foo }", "class Foo {\n #foo;\n}");
|
|
expectPrinted_("class Foo { #foo = 1 }", "class Foo {\n #foo = 1;\n}");
|
|
expectPrinted_("class Foo { #foo = #foo in this }", "class Foo {\n #foo = #foo in this;\n}");
|
|
expectPrinted_(
|
|
"class Foo { #foo = #foo in (#bar in this); #bar }",
|
|
"class Foo {\n #foo = #foo in (#bar in this);\n #bar;\n}",
|
|
);
|
|
expectPrinted_("class Foo { #foo() {} }", "class Foo {\n #foo() {\n }\n}");
|
|
expectPrinted_("class Foo { get #foo() {} }", "class Foo {\n get #foo() {\n }\n}");
|
|
expectPrinted_("class Foo { set #foo(x) {} }", "class Foo {\n set #foo(x) {\n }\n}");
|
|
expectPrinted_("class Foo { static #foo }", "class Foo {\n static #foo;\n}");
|
|
expectPrinted_("class Foo { static #foo = 1 }", "class Foo {\n static #foo = 1;\n}");
|
|
expectPrinted_("class Foo { static #foo() {} }", "class Foo {\n static #foo() {\n }\n}");
|
|
expectPrinted_("class Foo { static get #foo() {} }", "class Foo {\n static get #foo() {\n }\n}");
|
|
expectPrinted_("class Foo { static set #foo(x) {} }", "class Foo {\n static set #foo(x) {\n }\n}");
|
|
|
|
expectParseError("class Foo { #foo = #foo in #bar in this; #bar }", "Unexpected #bar");
|
|
|
|
expectParseError("class Foo { #constructor }", 'Invalid field name "#constructor"');
|
|
expectParseError("class Foo { #constructor() {} }", 'Invalid method name "#constructor"');
|
|
expectParseError("class Foo { static #constructor }", 'Invalid field name "#constructor"');
|
|
expectParseError("class Foo { static #constructor() {} }", 'Invalid method name "#constructor"');
|
|
expectParseError("class Foo { #\\u0063onstructor }", 'Invalid field name "#constructor"');
|
|
expectParseError("class Foo { #\\u0063onstructor() {} }", 'Invalid method name "#constructor"');
|
|
expectParseError("class Foo { static #\\u0063onstructor }", 'Invalid field name "#constructor"');
|
|
expectParseError("class Foo { static #\\u0063onstructor() {} }", 'Invalid method name "#constructor"');
|
|
const errorText = '"#foo" has already been declared';
|
|
expectParseError("class Foo { #foo; #foo }", errorText);
|
|
expectParseError("class Foo { #foo; static #foo }", errorText);
|
|
expectParseError("class Foo { static #foo; #foo }", errorText);
|
|
expectParseError("class Foo { #foo; #foo() {} }", errorText);
|
|
expectParseError("class Foo { #foo; get #foo() {} }", errorText);
|
|
expectParseError("class Foo { #foo; set #foo(x) {} }", errorText);
|
|
expectParseError("class Foo { #foo() {} #foo }", errorText);
|
|
expectParseError("class Foo { get #foo() {} #foo }", errorText);
|
|
expectParseError("class Foo { set #foo(x) {} #foo }", errorText);
|
|
expectParseError("class Foo { get #foo() {} get #foo() {} }", errorText);
|
|
expectParseError("class Foo { set #foo(x) {} set #foo(x) {} }", errorText);
|
|
expectParseError("class Foo { get #foo() {} set #foo(x) {} #foo }", errorText);
|
|
expectParseError("class Foo { set #foo(x) {} get #foo() {} #foo }", errorText);
|
|
|
|
expectPrinted_(
|
|
"class Foo { get #foo() {} set #foo(x) { this.#foo } }",
|
|
"class Foo {\n get #foo() {\n }\n set #foo(x) {\n this.#foo;\n }\n}",
|
|
);
|
|
expectPrinted_(
|
|
"class Foo { set #foo(x) { this.#foo } get #foo() {} }",
|
|
"class Foo {\n set #foo(x) {\n this.#foo;\n }\n get #foo() {\n }\n}",
|
|
);
|
|
expectPrinted_("class Foo { #foo } class Bar { #foo }", "class Foo {\n #foo;\n}\n\nclass Bar {\n #foo;\n}");
|
|
expectPrinted_("class Foo { foo = this.#foo; #foo }", "class Foo {\n foo = this.#foo;\n #foo;\n}");
|
|
expectPrinted_("class Foo { foo = this?.#foo; #foo }", "class Foo {\n foo = this?.#foo;\n #foo;\n}");
|
|
expectParseError(
|
|
"class Foo { #foo } class Bar { foo = this.#foo }",
|
|
'Private name "#foo" must be declared in an enclosing class',
|
|
);
|
|
expectParseError(
|
|
"class Foo { #foo } class Bar { foo = this?.#foo }",
|
|
'Private name "#foo" must be declared in an enclosing class',
|
|
);
|
|
expectParseError(
|
|
"class Foo { #foo } class Bar { foo = #foo in this }",
|
|
'Private name "#foo" must be declared in an enclosing class',
|
|
);
|
|
|
|
expectPrinted_(
|
|
`class Foo {
|
|
#if
|
|
#im() { return this.#im(this.#if) }
|
|
static #sf
|
|
static #sm() { return this.#sm(this.#sf) }
|
|
foo() {
|
|
return class {
|
|
#inner() {
|
|
return [this.#im, this?.#inner, this?.x.#if]
|
|
}
|
|
}
|
|
}
|
|
}
|
|
`,
|
|
`class Foo {
|
|
#if;
|
|
#im() {
|
|
return this.#im(this.#if);
|
|
}
|
|
static #sf;
|
|
static #sm() {
|
|
return this.#sm(this.#sf);
|
|
}
|
|
foo() {
|
|
return class {
|
|
#inner() {
|
|
return [this.#im, this?.#inner, this?.x.#if];
|
|
}
|
|
};
|
|
}
|
|
}`,
|
|
);
|
|
});
|
|
|
|
it("type only exports", () => {
|
|
let { expectPrinted_, expectParseError } = ts;
|
|
expectPrinted_("export type {foo, bar as baz} from 'bar'", "");
|
|
expectPrinted_("export type {foo, bar as baz}", "");
|
|
expectPrinted_("export type {foo} from 'bar'; x", "x");
|
|
expectPrinted_("export type {foo} from 'bar'\nx", "x");
|
|
expectPrinted_("export type {default} from 'bar'", "");
|
|
expectPrinted_("export { type } from 'mod'; type", 'export { type } from "mod";\ntype');
|
|
expectPrinted_("export { type, as } from 'mod'", 'export { type, as } from "mod"');
|
|
expectPrinted_("export { x, type foo } from 'mod'; x", 'export { x } from "mod";\nx');
|
|
expectPrinted_("export { x, type as } from 'mod'; x", 'export { x } from "mod";\nx');
|
|
expectPrinted_("export { x, type foo as bar } from 'mod'; x", 'export { x } from "mod";\nx');
|
|
expectPrinted_("export { x, type foo as as } from 'mod'; x", 'export { x } from "mod";\nx');
|
|
expectPrinted_("export { type as as } from 'mod'; as", 'export { type as as } from "mod";\nas');
|
|
expectPrinted_("export { type as foo } from 'mod'; foo", 'export { type as foo } from "mod";\nfoo');
|
|
expectPrinted_("export { type as type } from 'mod'; type", 'export { type } from "mod";\ntype');
|
|
expectPrinted_("export { x, type as as foo } from 'mod'; x", 'export { x } from "mod";\nx');
|
|
expectPrinted_("export { x, type as as as } from 'mod'; x", 'export { x } from "mod";\nx');
|
|
expectPrinted_("export { x, type type as as } from 'mod'; x", 'export { x } from "mod";\nx');
|
|
expectPrinted_("export { x, \\u0074ype y }; let x, y", "export { x };\nlet x, y");
|
|
expectPrinted_("export { x, \\u0074ype y } from 'mod'", 'export { x } from "mod"');
|
|
expectPrinted_("export { x, type if } from 'mod'", 'export { x } from "mod"');
|
|
expectPrinted_("export { x, type y as if }; let x", "export { x };\nlet x");
|
|
expectPrinted_("export { type x };", "");
|
|
});
|
|
|
|
it("delete + optional chain", () => {
|
|
expectPrinted_("delete foo.bar.baz", "delete foo.bar.baz");
|
|
expectPrinted_("delete foo?.bar.baz", "delete foo?.bar.baz");
|
|
expectPrinted_("delete foo?.bar?.baz", "delete foo?.bar?.baz");
|
|
});
|
|
|
|
it("useDefineForConst TypeScript class initialization", () => {
|
|
var { expectPrinted_ } = ts;
|
|
expectPrinted_(
|
|
`
|
|
class Foo {
|
|
constructor(public x: string = "hey") {}
|
|
bar: number;
|
|
}
|
|
`.trim(),
|
|
`
|
|
class Foo {
|
|
x;
|
|
constructor(x = "hey") {
|
|
this.x = x;
|
|
}
|
|
bar;
|
|
}
|
|
`.trim(),
|
|
);
|
|
});
|
|
|
|
it("class static blocks", () => {
|
|
expectPrinted_("class Foo { static {} }", "class Foo {\n static {\n }\n}");
|
|
expectPrinted_("class Foo { static {} x = 1 }", "class Foo {\n static {\n }\n x = 1;\n}");
|
|
expectPrinted_("class Foo { static { this.foo() } }", "class Foo {\n static {\n this.foo();\n }\n}");
|
|
|
|
expectParseError("class Foo { static { yield } }", '"yield" is a reserved word and cannot be used in strict mode');
|
|
expectParseError("class Foo { static { await } }", 'The keyword "await" cannot be used here');
|
|
expectParseError("class Foo { static { return } }", "A return statement cannot be used here");
|
|
expectParseError("class Foo { static { break } }", 'Cannot use "break" here');
|
|
expectParseError("class Foo { static { continue } }", 'Cannot use "continue" here');
|
|
expectParseError("x: { class Foo { static { break x } } }", 'There is no containing label named "x"');
|
|
expectParseError("x: { class Foo { static { continue x } } }", 'There is no containing label named "x"');
|
|
|
|
expectParseError("class Foo { get #x() { this.#x = 1 } }", 'Writing to getter-only property "#x" will throw');
|
|
expectParseError("class Foo { get #x() { this.#x += 1 } }", 'Writing to getter-only property "#x" will throw');
|
|
expectParseError("class Foo { set #x(x) { this.#x } }", 'Reading from setter-only property "#x" will throw');
|
|
expectParseError("class Foo { set #x(x) { this.#x += 1 } }", 'Reading from setter-only property "#x" will throw');
|
|
|
|
// Writing to method warnings
|
|
expectParseError("class Foo { #x() { this.#x = 1 } }", 'Writing to read-only method "#x" will throw');
|
|
expectParseError("class Foo { #x() { this.#x += 1 } }", 'Writing to read-only method "#x" will throw');
|
|
});
|
|
|
|
describe("simplification", () => {
|
|
const transpiler = new Bun.Transpiler({
|
|
loader: "tsx",
|
|
define: {
|
|
"process.env.NODE_ENV": JSON.stringify("development"),
|
|
user_undefined: "undefined",
|
|
},
|
|
macro: {
|
|
react: {
|
|
bacon: `${import.meta.dir}/macro-check.js`,
|
|
},
|
|
},
|
|
platform: "browser",
|
|
minify: { syntax: true },
|
|
});
|
|
|
|
const parsed = (code, trim = true, autoExport = false, transpiler_ = transpiler) => {
|
|
if (autoExport) {
|
|
code = "export default (" + code + ")";
|
|
}
|
|
|
|
var out = transpiler_.transformSync(code, "js");
|
|
if (autoExport && out.startsWith("export default ")) {
|
|
out = out.substring("export default ".length);
|
|
}
|
|
|
|
if (trim) {
|
|
out = out.trim();
|
|
|
|
if (out.endsWith(";")) {
|
|
out = out.substring(0, out.length - 1);
|
|
}
|
|
|
|
return out.trim();
|
|
}
|
|
|
|
return out;
|
|
};
|
|
|
|
const expectPrinted = (code, out) => {
|
|
expect(parsed(code, true, true)).toBe(out);
|
|
};
|
|
|
|
const expectPrinted_ = (code, out) => {
|
|
expect(parsed(code, !out.endsWith(";\n"), false)).toBe(out);
|
|
};
|
|
|
|
const expectPrintedNoTrim = (code, out) => {
|
|
expect(parsed(code, false, false)).toBe(out);
|
|
};
|
|
|
|
const expectBunPrinted_ = (code, out) => {
|
|
expect(parsed(code, !out.endsWith(";\n"), false, bunTranspiler)).toBe(out);
|
|
};
|
|
|
|
it("unary operator", () => {
|
|
expectPrinted("a = !(b, c)", "a = (b, !c)");
|
|
});
|
|
|
|
it.todo("const inlining", () => {
|
|
var transpiler = new Bun.Transpiler({
|
|
inline: true,
|
|
platform: "bun",
|
|
allowBunRuntime: false,
|
|
minify: { syntax: true },
|
|
});
|
|
|
|
function check(input, output) {
|
|
expect(
|
|
transpiler
|
|
.transformSync("export function hello() {\n" + input + "\n}")
|
|
.trim()
|
|
.replaceAll(/^ /gm, ""),
|
|
).toBe("export function hello() {\n" + output + "\n}".replaceAll(/^ /gm, ""));
|
|
}
|
|
hideFromStackTrace(check);
|
|
|
|
check("const x = 1; return x", "return 1;");
|
|
check("const x = 1; return x + 1", "return 2;");
|
|
check("const x = 1; return x + x", "return 2;");
|
|
check("const x = 1; return x + x + 1", "return 3;");
|
|
check("const x = 1; return x + x + x", "return 3;");
|
|
check(`const foo = "foo"; const bar = "bar"; return foo + bar`, `return "foobar";`);
|
|
|
|
check(
|
|
`
|
|
const a = "a";
|
|
const c = "b" + a;
|
|
const b = c + a;
|
|
const d = b + a;
|
|
console.log(a, b, c, d);
|
|
`,
|
|
`
|
|
const c = "ba", b = c + "a", d = b + "a";
|
|
console.log("a", b, c, d);
|
|
`.trim(),
|
|
);
|
|
|
|
// check that it doesn't inline after "var"
|
|
check(
|
|
`
|
|
const x = 1;
|
|
const y = 2;
|
|
var hey = "yo";
|
|
const z = 3;
|
|
console.log(x + y + z);
|
|
`,
|
|
`
|
|
var hey = "yo";
|
|
const z = 3;
|
|
console.log(3 + z);
|
|
`.trim(),
|
|
);
|
|
|
|
// check that nested scopes can inline from parent scopes
|
|
check(
|
|
`
|
|
const x = 1;
|
|
const y = 2;
|
|
var hey = "yo";
|
|
const z = 3;
|
|
function hey() {
|
|
const boom = 3;
|
|
return x + y + boom + hey;
|
|
}
|
|
hey();
|
|
`,
|
|
`
|
|
var hey = "yo";
|
|
const z = 3;
|
|
function hey() {
|
|
return 6 + hey;
|
|
}
|
|
hey();
|
|
`.trim(),
|
|
);
|
|
|
|
// check that we don't inline objects or arrays that aren't from macros
|
|
check(
|
|
`
|
|
const foo = { bar: true };
|
|
const array = [1];
|
|
console.log(foo, array);
|
|
`,
|
|
`
|
|
const foo = { bar: !0 }, array = [1];
|
|
console.log(foo, array);
|
|
`.trim(),
|
|
);
|
|
});
|
|
|
|
it("constant folding scopes", () => {
|
|
var transpiler = new Bun.Transpiler({
|
|
inline: true,
|
|
platform: "bun",
|
|
allowBunRuntime: false,
|
|
minify: { syntax: true },
|
|
});
|
|
|
|
// Check that pushing/popping scopes doesn't cause a crash
|
|
// We panic at runtime if the scopes are unbalanced, so this test just checks we don't have any crashes
|
|
function check(input) {
|
|
transpiler.transformSync(input);
|
|
}
|
|
|
|
check("var x; 1 ? 0 : ()=>{}; (()=>{})()");
|
|
check("var x; 0 ? ()=>{} : 1; (()=>{})()");
|
|
check("var x; 0 && (()=>{}); (()=>{})()");
|
|
check("var x; 1 || (()=>{}); (()=>{})()");
|
|
check("if (1) 0; else ()=>{}; (()=>{})()");
|
|
check("if (0) ()=>{}; else 1; (()=>{})()");
|
|
check(`
|
|
var func = () => {};
|
|
var x;
|
|
1 ? 0 : func;
|
|
(() => {})();
|
|
switch (1) {
|
|
case 0: {
|
|
class Foo {
|
|
static {
|
|
function hey() {
|
|
return class {
|
|
static {
|
|
var foo = class {
|
|
hey(arg) {
|
|
return 1;
|
|
}
|
|
};
|
|
new foo();
|
|
}
|
|
};
|
|
}
|
|
}
|
|
}
|
|
new Foo();
|
|
}
|
|
}
|
|
`);
|
|
});
|
|
|
|
it.todo("substitution", () => {
|
|
var transpiler = new Bun.Transpiler({
|
|
inline: true,
|
|
platform: "bun",
|
|
allowBunRuntime: false,
|
|
minify: { syntax: true },
|
|
});
|
|
function check(input, output) {
|
|
expect(
|
|
transpiler
|
|
.transformSync("export function hello() {\n" + input + "\n}")
|
|
.trim()
|
|
.replaceAll(/^ /gm, ""),
|
|
).toBe("export function hello() {\n" + output + "\n}".replaceAll(/^ /gm, ""));
|
|
}
|
|
check("var x = 1; return x", "var x = 1;\nreturn x;");
|
|
check("let x = 1; return x", "return 1;");
|
|
check("const x = 1; return x", "return 1;");
|
|
|
|
check("let x = 1; if (false) x++; return x", "return 1;");
|
|
// TODO: comma operator
|
|
// check("let x = 1; if (true) x++; return x", "let x = 1;\nreturn x++, x;");
|
|
check("let x = 1; return x + x", "let x = 1;\nreturn x + x;");
|
|
|
|
// Can substitute into normal unary operators
|
|
check("let x = 1; return +x", "return 1;");
|
|
check("let x = 1; return -x", "return -1;");
|
|
check("let x = 1; return !x", "return !1;");
|
|
check("let x = 1; return ~x", "return ~1;");
|
|
// TODO: remove needless return undefined;
|
|
// check("let x = 1; return void x", "let x = 1;");
|
|
|
|
// esbuild does this:
|
|
// check("let x = 1; return typeof x", "return typeof 1;");
|
|
// we do:
|
|
check("let x = 1; return typeof x", 'return "number";');
|
|
|
|
// Check substituting a side-effect free value into normal binary operators
|
|
// esbuild does this:
|
|
// check("let x = 1; return x + 2", "return 1 + 2;");
|
|
// we do:
|
|
check("let x = 1; return x + 2", "return 3;");
|
|
check("let x = 1; return 2 + x", "return 3;");
|
|
check("let x = 1; return x + arg0", "return 1 + arg0;");
|
|
// check("let x = 1; return arg0 + x", "return arg0 + 1;");
|
|
check("let x = 1; return x + fn()", "return 1 + fn();");
|
|
check("let x = 1; return fn() + x", "let x = 1;\nreturn fn() + x;");
|
|
check("let x = 1; return x + undef", "return 1 + undef;");
|
|
check("let x = 1; return undef + x", "let x = 1;\nreturn undef + x;");
|
|
|
|
// Check substituting a value with side-effects into normal binary operators
|
|
check("let x = fn(); return x + 2", "return fn() + 2;");
|
|
check("let x = fn(); return 2 + x", "return 2 + fn();");
|
|
check("let x = fn(); return x + arg0", "return fn() + arg0;");
|
|
check("let x = fn(); return arg0 + x", "let x = fn();\nreturn arg0 + x;");
|
|
check("let x = fn(); return x + fn2()", "return fn() + fn2();");
|
|
check("let x = fn(); return fn2() + x", "let x = fn();\nreturn fn2() + x;");
|
|
check("let x = fn(); return x + undef", "return fn() + undef;");
|
|
check("let x = fn(); return undef + x", "let x = fn();\nreturn undef + x;");
|
|
|
|
// Cannot substitute into mutating unary operators
|
|
check("let x = 1; ++x", "let x = 1;\n++x;");
|
|
check("let x = 1; --x", "let x = 1;\n--x;");
|
|
check("let x = 1; x++", "let x = 1;\nx++;");
|
|
check("let x = 1; x--", "let x = 1;\nx--;");
|
|
check("let x = 1; delete x", "let x = 1;\ndelete x;");
|
|
|
|
// Cannot substitute into mutating binary operators unless immediately after the assignment
|
|
check("let x = 1; let y; x = 2", "let x = 1, y;\nx = 2;");
|
|
check("let x = 1; let y; x += 2", "let x = 1, y;\nx += 2;");
|
|
check("let x = 1; let y; x ||= 2", "let x = 1, y;\nx ||= 2;");
|
|
|
|
// Can substitute past mutating binary operators when the left operand has no side effects
|
|
// check("let x = 1; arg0 = x", "arg0 = 1;");
|
|
// check("let x = 1; arg0 += x", "arg0 += 1;");
|
|
// check("let x = 1; arg0 ||= x", "arg0 ||= 1;");
|
|
// check("let x = fn(); arg0 = x", "arg0 = fn();");
|
|
// check("let x = fn(); arg0 += x", "let x = fn();\narg0 += x;");
|
|
// check("let x = fn(); arg0 ||= x", "let x = fn();\narg0 ||= x;");
|
|
|
|
// Cannot substitute past mutating binary operators when the left operand has side effects
|
|
check("let x = 1; y.z = x", "let x = 1;\ny.z = x;");
|
|
check("let x = 1; y.z += x", "let x = 1;\ny.z += x;");
|
|
check("let x = 1; y.z ||= x", "let x = 1;\ny.z ||= x;");
|
|
check("let x = fn(); y.z = x", "let x = fn();\ny.z = x;");
|
|
check("let x = fn(); y.z += x", "let x = fn();\ny.z += x;");
|
|
check("let x = fn(); y.z ||= x", "let x = fn();\ny.z ||= x;");
|
|
|
|
// TODO:
|
|
// Can substitute code without side effects into branches
|
|
// check("let x = arg0; return x ? y : z;", "return arg0 ? y : z;");
|
|
// check("let x = arg0; return arg1 ? x : y;", "return arg1 ? arg0 : y;");
|
|
// check("let x = arg0; return arg1 ? y : x;", "return arg1 ? y : arg0;");
|
|
// check("let x = arg0; return x || y;", "return arg0 || y;");
|
|
// check("let x = arg0; return x && y;", "return arg0 && y;");
|
|
// check("let x = arg0; return x ?? y;", "return arg0 ?? y;");
|
|
// check("let x = arg0; return arg1 || x;", "return arg1 || arg0;");
|
|
// check("let x = arg0; return arg1 && x;", "return arg1 && arg0;");
|
|
// check("let x = arg0; return arg1 ?? x;", "return arg1 ?? arg0;");
|
|
|
|
// Can substitute code without side effects into branches past an expression with side effects
|
|
// check(
|
|
// "let x = arg0; return y ? x : z;",
|
|
// "let x = arg0;\nreturn y ? x : z;",
|
|
// );
|
|
// check(
|
|
// "let x = arg0; return y ? z : x;",
|
|
// "let x = arg0;\nreturn y ? z : x;",
|
|
// );
|
|
// check("let x = arg0; return (arg1 ? 1 : 2) ? x : 3;", "return arg0;");
|
|
// check(
|
|
// "let x = arg0; return (arg1 ? 1 : 2) ? 3 : x;",
|
|
// "let x = arg0;\nreturn 3;",
|
|
// );
|
|
// check(
|
|
// "let x = arg0; return (arg1 ? y : 1) ? x : 2;",
|
|
// "let x = arg0;\nreturn !arg1 || y ? x : 2;",
|
|
// );
|
|
// check(
|
|
// "let x = arg0; return (arg1 ? 1 : y) ? x : 2;",
|
|
// "let x = arg0;\nreturn arg1 || y ? x : 2;",
|
|
// );
|
|
// check(
|
|
// "let x = arg0; return (arg1 ? y : 1) ? 2 : x;",
|
|
// "let x = arg0;\nreturn !arg1 || y ? 2 : x;",
|
|
// );
|
|
// check(
|
|
// "let x = arg0; return (arg1 ? 1 : y) ? 2 : x;",
|
|
// "let x = arg0;\nreturn arg1 || y ? 2 : x;",
|
|
// );
|
|
// check("let x = arg0; return y || x;", "let x = arg0;\nreturn y || x;");
|
|
// check("let x = arg0; return y && x;", "let x = arg0;\nreturn y && x;");
|
|
// check("let x = arg0; return y ?? x;", "let x = arg0;\nreturn y ?? x;");
|
|
|
|
// Cannot substitute code with side effects into branches
|
|
check("let x = fn(); return x ? arg0 : y;", "return fn() ? arg0 : y;");
|
|
check("let x = fn(); return arg0 ? x : y;", "let x = fn();\nreturn arg0 ? x : y;");
|
|
check("let x = fn(); return arg0 ? y : x;", "let x = fn();\nreturn arg0 ? y : x;");
|
|
check("let x = fn(); return x || arg0;", "return fn() || arg0;");
|
|
check("let x = fn(); return x && arg0;", "return fn() && arg0;");
|
|
check("let x = fn(); return x ?? arg0;", "return fn() ?? arg0;");
|
|
check("let x = fn(); return arg0 || x;", "let x = fn();\nreturn arg0 || x;");
|
|
check("let x = fn(); return arg0 && x;", "let x = fn();\nreturn arg0 && x;");
|
|
check("let x = fn(); return arg0 ?? x;", "let x = fn();\nreturn arg0 ?? x;");
|
|
|
|
// Test chaining
|
|
check("let x = fn(); let y = x[prop]; let z = y.val; throw z", "throw fn()[prop].val;");
|
|
check("let x = fn(), y = x[prop], z = y.val; throw z", "throw fn()[prop].val;");
|
|
|
|
// Can substitute an initializer with side effects
|
|
check("let x = 0; let y = ++x; return y", "let x = 0;\nreturn ++x;");
|
|
|
|
// Can substitute an initializer without side effects past an expression without side effects
|
|
check("let x = 0; let y = x; return [x, y]", "let x = 0;\nreturn [x, x];");
|
|
|
|
// TODO: merge s_local
|
|
// Cannot substitute an initializer with side effects past an expression without side effects
|
|
// check(
|
|
// "let x = 0; let y = ++x; return [x, y]",
|
|
// "let x = 0, y = ++x;\nreturn [x, y];",
|
|
// );
|
|
|
|
// Cannot substitute an initializer without side effects past an expression with side effects
|
|
// TODO: merge s_local
|
|
// check(
|
|
// "let x = 0; let y = {valueOf() { x = 1 }}; let z = x; return [y == 1, z]",
|
|
// "let x = 0, y = { valueOf() {\n x = 1;\n} }, z = x;\nreturn [y == 1, z];",
|
|
// );
|
|
|
|
// Cannot inline past a spread operator, since that evaluates code
|
|
check("let x = arg0; return [...x];", "return [...arg0];");
|
|
check("let x = arg0; return [x, ...arg1];", "return [arg0, ...arg1];");
|
|
check("let x = arg0; return [...arg1, x];", "let x = arg0;\nreturn [...arg1, x];");
|
|
// TODO: preserve call here
|
|
// check("let x = arg0; return arg1(...x);", "return arg1(...arg0);");
|
|
// check(
|
|
// "let x = arg0; return arg1(x, ...arg1);",
|
|
// "return arg1(arg0, ...arg1);",
|
|
// );
|
|
check("let x = arg0; return arg1(...arg1, x);", "let x = arg0;\nreturn arg1(...arg1, x);");
|
|
|
|
// Test various statement kinds
|
|
// TODO:
|
|
// check("let x = arg0; arg1(x);", "arg1(arg0);");
|
|
|
|
check("let x = arg0; throw x;", "throw arg0;");
|
|
check("let x = arg0; return x;", "return arg0;");
|
|
check("let x = arg0; if (x) return 1;", "if (arg0)\n return 1;");
|
|
check("let x = arg0; switch (x) { case 0: return 1; }", "switch (arg0) {\n case 0:\n return 1;\n}");
|
|
check("let x = arg0; let y = x; return y + y;", "let y = arg0;\nreturn y + y;");
|
|
|
|
// Loops must not be substituted into because they evaluate multiple times
|
|
check("let x = arg0; do {} while (x);", "let x = arg0;\ndo\n ;\nwhile (x);");
|
|
|
|
// TODO: convert while(x) to for (;x;)
|
|
check(
|
|
"let x = arg0; while (x) return 1;",
|
|
"let x = arg0;\nwhile (x)\n return 1;",
|
|
// "let x = arg0;\nfor (; x; )\n return 1;",
|
|
);
|
|
check("let x = arg0; for (; x; ) return 1;", "let x = arg0;\nfor (;x; )\n return 1;");
|
|
|
|
// Can substitute an expression without side effects into a branch due to optional chaining
|
|
// TODO:
|
|
// check("let x = arg0; return arg1?.[x];", "return arg1?.[arg0];");
|
|
// check("let x = arg0; return arg1?.(x);", "return arg1?.(arg0);");
|
|
|
|
// Cannot substitute an expression with side effects into a branch due to optional chaining,
|
|
// since that would change the expression with side effects from being unconditionally
|
|
// evaluated to being conditionally evaluated, which is a behavior change
|
|
check("let x = fn(); return arg1?.[x];", "let x = fn();\nreturn arg1?.[x];");
|
|
check("let x = fn(); return arg1?.(x);", "let x = fn();\nreturn arg1?.(x);");
|
|
|
|
// Can substitute an expression past an optional chaining operation, since it has side effects
|
|
check("let x = arg0; return arg1?.a === x;", "let x = arg0;\nreturn arg1?.a === x;");
|
|
check("let x = arg0; return arg1?.[0] === x;", "let x = arg0;\nreturn arg1?.[0] === x;");
|
|
check("let x = arg0; return arg1?.(0) === x;", "let x = arg0;\nreturn arg1?.(0) === x;");
|
|
check("let x = arg0; return arg1?.a[x];", "let x = arg0;\nreturn arg1?.a[x];");
|
|
check("let x = arg0; return arg1?.a(x);", "let x = arg0;\nreturn arg1?.a(x);");
|
|
// TODO:
|
|
// check(
|
|
// "let x = arg0; return arg1?.[a][x];",
|
|
// "let x = arg0;\nreturn arg1?.[a][x];",
|
|
// );
|
|
check("let x = arg0; return arg1?.[a](x);", "let x = arg0;\nreturn (arg1?.[a])(x);");
|
|
check("let x = arg0; return arg1?.(a)[x];", "let x = arg0;\nreturn (arg1?.(a))[x];");
|
|
check("let x = arg0; return arg1?.(a)(x);", "let x = arg0;\nreturn (arg1?.(a))(x);");
|
|
|
|
// Can substitute into an object as long as there are no side effects
|
|
// beforehand. Note that computed properties must call "toString()" which
|
|
// can have side effects.
|
|
check("let x = arg0; return {x};", "return { x: arg0 };");
|
|
check("let x = arg0; return {x: y, y: x};", "let x = arg0;\nreturn { x: y, y: x };");
|
|
// TODO:
|
|
// check(
|
|
// "let x = arg0; return {x: arg1, y: x};",
|
|
// "return { x: arg1, y: arg0 };",
|
|
// );
|
|
check("let x = arg0; return {[x]: 0};", "return { [arg0]: 0 };");
|
|
check("let x = arg0; return {[y]: x};", "let x = arg0;\nreturn { [y]: x };");
|
|
check("let x = arg0; return {[arg1]: x};", "let x = arg0;\nreturn { [arg1]: x };");
|
|
// TODO:
|
|
// check(
|
|
// "let x = arg0; return {y() {}, x};",
|
|
// "return { y() {\n}, x: arg0 };",
|
|
// );
|
|
check("let x = arg0; return {[y]() {}, x};", "let x = arg0;\nreturn { [y]() {\n}, x };");
|
|
check("let x = arg0; return {...x};", "return { ...arg0 };");
|
|
check("let x = arg0; return {...x, y};", "return { ...arg0, y };");
|
|
check("let x = arg0; return {x, ...y};", "return { x: arg0, ...y };");
|
|
check("let x = arg0; return {...y, x};", "let x = arg0;\nreturn { ...y, x };");
|
|
|
|
// TODO:
|
|
// Check substitutions into template literals
|
|
// check("let x = arg0; return `a${x}b${y}c`;", "return `a${arg0}b${y}c`;");
|
|
// check(
|
|
// "let x = arg0; return `a${y}b${x}c`;",
|
|
// "let x = arg0;\nreturn `a${y}b${x}c`;",
|
|
// );
|
|
// check(
|
|
// "let x = arg0; return `a${arg1}b${x}c`;",
|
|
// "return `a${arg1}b${arg0}c`;",
|
|
// );
|
|
// check("let x = arg0; return x`y`;", "return arg0`y`;");
|
|
// check(
|
|
// "let x = arg0; return y`a${x}b`;",
|
|
// "let x = arg0;\nreturn y`a${x}b`;",
|
|
// );
|
|
// check("let x = arg0; return arg1`a${x}b`;", "return arg1`a${arg0}b`;");
|
|
// check("let x = 'x'; return `a${x}b`;", "return `axb`;");
|
|
|
|
// Check substitutions into import expressions
|
|
// TODO:
|
|
// check("let x = arg0; return import(x);", "return import(arg0);");
|
|
// check(
|
|
// "let x = arg0; return [import(y), x];",
|
|
// "let x = arg0;\nreturn [import(y), x];",
|
|
// );
|
|
// check(
|
|
// "let x = arg0; return [import(arg1), x];",
|
|
// "return [import(arg1), arg0];",
|
|
// );
|
|
|
|
// Check substitutions into await expressions
|
|
check("return async () => { let x = arg0; await x; };", "return async () => {\n await arg0;\n};");
|
|
|
|
// TODO: combine with comma operator
|
|
// check(
|
|
// "return async () => { let x = arg0; await y; return x; };",
|
|
// "return async () => {\n let x = arg0;\n return await y, x;\n};",
|
|
// );
|
|
// check(
|
|
// "return async () => { let x = arg0; await arg1; return x; };",
|
|
// "return async () => {\n let x = arg0;\n return await arg1, x;\n};",
|
|
// );
|
|
|
|
// Check substitutions into yield expressions
|
|
check("return function* () { let x = arg0; yield x; };", "return function* () {\n yield arg0;\n};");
|
|
// TODO: combine with comma operator
|
|
// check(
|
|
// "return function* () { let x = arg0; yield; return x; };",
|
|
// "return function* () {\n let x = arg0;\n yield ; \n return x;\n};",
|
|
// );
|
|
// check(
|
|
// "return function* () { let x = arg0; yield y; return x; };",
|
|
// "return function* () {\n let x = arg0;\n return yield y, x;\n};",
|
|
// );
|
|
// check(
|
|
// "return function* () { let x = arg0; yield arg1; return x; };",
|
|
// "return function* () {\n let x = arg0;\n return yield arg1, x;\n};",
|
|
// );
|
|
|
|
// Cannot substitute into call targets when it would change "this"
|
|
check("let x = arg0; x()", "arg0();");
|
|
// check("let x = arg0; (0, x)()", "arg0();");
|
|
check("let x = arg0.foo; x.bar()", "arg0.foo.bar();");
|
|
check("let x = arg0.foo; x[bar]()", "arg0.foo[bar]();");
|
|
check("let x = arg0.foo; x()", "let x = arg0.foo;\nx();");
|
|
check("let x = arg0[foo]; x()", "let x = arg0[foo];\nx();");
|
|
check("let x = arg0?.foo; x()", "let x = arg0?.foo;\nx();");
|
|
check("let x = arg0?.[foo]; x()", "let x = arg0?.[foo];\nx();");
|
|
// check("let x = arg0.foo; (0, x)()", "let x = arg0.foo;\nx();");
|
|
// check("let x = arg0[foo]; (0, x)()", "let x = arg0[foo];\nx();");
|
|
// check("let x = arg0?.foo; (0, x)()", "let x = arg0?.foo;\nx();");
|
|
// check("let x = arg0?.[foo]; (0, x)()", "let x = arg0?.[foo];\nx();");
|
|
});
|
|
|
|
it("constant folding", () => {
|
|
// we have an optimization for numbers 0 - 100, -0 - -100 so we must test those specifically
|
|
// https://github.com/oven-sh/bun/issues/2810
|
|
for (let i = 1; i < 120; i++) {
|
|
const inner = "${" + i + " * 1}";
|
|
expectPrinted("console.log(`" + inner + "`)", 'console.log("' + i + '")');
|
|
|
|
const innerNeg = "${" + -i + " * 1}";
|
|
expectPrinted("console.log(`" + innerNeg + "`)", 'console.log("' + -i + '")');
|
|
}
|
|
|
|
expectPrinted("1 && 2", "2");
|
|
expectPrinted("1 || 2", "1");
|
|
expectPrinted("0 && 1", "0");
|
|
expectPrinted("0 || 1", "1");
|
|
|
|
expectPrinted("null ?? 1", "1");
|
|
expectPrinted("undefined ?? 1", "1");
|
|
expectPrinted("0 ?? 1", "0");
|
|
expectPrinted("false ?? 1", "!1");
|
|
expectPrinted('"" ?? 1', '""');
|
|
|
|
expectPrinted("typeof undefined", '"undefined"');
|
|
expectPrinted("typeof null", '"object"');
|
|
expectPrinted("typeof false", '"boolean"');
|
|
expectPrinted("typeof true", '"boolean"');
|
|
expectPrinted("typeof 123", '"number"');
|
|
expectPrinted("typeof 123n", '"bigint"');
|
|
expectPrinted("typeof 'abc'", '"string"');
|
|
expectPrinted("typeof function() {}", '"function"');
|
|
expectPrinted("typeof (() => {})", '"function"');
|
|
expectPrinted("typeof {}", '"object"');
|
|
expectPrinted("typeof {foo: 123}", '"object"');
|
|
expectPrinted("typeof []", '"object"');
|
|
expectPrinted("typeof [0]", '"object"');
|
|
expectPrinted("typeof [null]", '"object"');
|
|
expectPrinted("typeof ['boolean']", '"object"');
|
|
|
|
expectPrinted('typeof [] === "object"', "!0");
|
|
expectPrinted("typeof {foo: 123} === typeof {bar: 123}", "!0");
|
|
expectPrinted("typeof {foo: 123} !== typeof 123", "!0");
|
|
|
|
expectPrinted("undefined === undefined", "!0");
|
|
expectPrinted("undefined !== undefined", "!1");
|
|
expectPrinted("undefined == undefined", "!0");
|
|
expectPrinted("undefined != undefined", "!1");
|
|
|
|
expectPrinted("null === null", "!0");
|
|
expectPrinted("null !== null", "!1");
|
|
expectPrinted("null == null", "!0");
|
|
expectPrinted("null != null", "!1");
|
|
|
|
expectPrinted("undefined === null", "!1");
|
|
expectPrinted("undefined !== null", "!0");
|
|
expectPrinted("undefined == null", "!0");
|
|
expectPrinted("undefined != null", "!1");
|
|
|
|
expectPrinted("true === true", "!0");
|
|
expectPrinted("true === false", "!1");
|
|
expectPrinted("true !== true", "!1");
|
|
expectPrinted("true !== false", "!0");
|
|
expectPrinted("true == true", "!0");
|
|
expectPrinted("true == false", "!1");
|
|
expectPrinted("true != true", "!1");
|
|
expectPrinted("true != false", "!0");
|
|
|
|
expectPrinted("1 === 1", "!0");
|
|
expectPrinted("1 === 2", "!1");
|
|
expectPrinted("1 === '1'", '1 === "1"');
|
|
expectPrinted("1 == 1", "!0");
|
|
expectPrinted("1 == 2", "!1");
|
|
expectPrinted("1 == '1'", '1 == "1"');
|
|
|
|
expectPrinted("1 !== 1", "!1");
|
|
expectPrinted("1 !== 2", "!0");
|
|
expectPrinted("1 !== '1'", '1 !== "1"');
|
|
expectPrinted("1 != 1", "!1");
|
|
expectPrinted("1 != 2", "!0");
|
|
expectPrinted("1 != '1'", '1 != "1"');
|
|
|
|
expectPrinted('"" == 0', "!0");
|
|
expectPrinted("1n == 1n", "!0");
|
|
expectPrinted("1234n == 1234n", "!0");
|
|
expectPrinted("0x00n == 0n", "0x00n == 0n");
|
|
expectPrinted("1n == 2n", "1n == 2n");
|
|
|
|
expectPrinted("'a' === '\\x61'", "!0");
|
|
expectPrinted("'a' === '\\x62'", "!1");
|
|
expectPrinted("'a' === 'abc'", "!1");
|
|
expectPrinted("'a' !== '\\x61'", "!1");
|
|
expectPrinted("'a' !== '\\x62'", "!0");
|
|
expectPrinted("'a' !== 'abc'", "!0");
|
|
expectPrinted("'a' == '\\x61'", "!0");
|
|
expectPrinted("'a' == '\\x62'", "!1");
|
|
expectPrinted("'a' == 'abc'", "!1");
|
|
expectPrinted("'a' != '\\x61'", "!1");
|
|
expectPrinted("'a' != '\\x62'", "!0");
|
|
expectPrinted("'a' != 'abc'", "!0");
|
|
|
|
expectPrinted("'a' + 'b'", '"ab"');
|
|
expectPrinted("'a' + 'bc'", '"abc"');
|
|
expectPrinted("'ab' + 'c'", '"abc"');
|
|
expectPrinted("x + 'a' + 'b'", 'x + "ab"');
|
|
expectPrinted("x + 'a' + 'bc'", 'x + "abc"');
|
|
expectPrinted("x + 'ab' + 'c'", 'x + "abc"');
|
|
expectPrinted("'a' + 1", '"a" + 1');
|
|
expectPrinted("x * 'a' + 'b'", 'x * "a" + "b"');
|
|
|
|
// rope string push another rope string
|
|
expectPrinted("'a' + ('b' + 'c') + 'd'", '"abcd"');
|
|
expectPrinted("('a' + 'b') + 'c'", '"abc"');
|
|
expectPrinted("'a' + ('b' + 'c')", '"abc"');
|
|
expectPrinted("'a' + ('b' + ('c' + 'd')) + 'e'", '"abcde"');
|
|
expectPrinted("'a' + ('b' + ('c' + ('d' + 'e')))", '"abcde"');
|
|
expectPrinted("('a' + ('b' + ('c' + 'd'))) + 'e'", '"abcde"');
|
|
expectPrinted("('a' + ('b' + 'c')) + ('d' + 'e')", '"abcde"');
|
|
expectPrinted("('a' + 'b') + ('c' + 'd')", '"abcd"');
|
|
expectPrinted("'a' + ('b' + 'c') + 'd'", '"abcd"');
|
|
expectPrinted("'a' + ('b' + ('c' + 'd'))", '"abcd"');
|
|
|
|
function check(input, output) {
|
|
expect(transpiler.transformSync(input)).toEqual(output);
|
|
}
|
|
|
|
var output = `var boop = "bcd";\nconst ropy = "a" + boop + "d", ropy2 = "b" + boop;\n`;
|
|
check(`var boop = ('b' + 'c') + 'd'; const ropy = "a" + boop + 'd'; const ropy2 = 'b' + boop;`, output);
|
|
|
|
output = `var boop = "fbcd", ropy = "a" + boop + "d", ropy2 = "b" + (ropy + "d");\n`;
|
|
check(`var boop = "f" + ("b" + "c") + "d";var ropy = "a" + boop + "d";var ropy2 = "b" + (ropy + "d")`, output);
|
|
|
|
expectPrinted("'string' + `template`", `"stringtemplate"`);
|
|
|
|
expectPrinted("`template` + 'string'", '"templatestring"');
|
|
|
|
// TODO: string template simplification
|
|
// expectPrinted("'string' + `a${foo}b`", "`stringa${foo}b`");
|
|
// expectPrinted("'string' + tag`template`", '"string" + tag`template`;');
|
|
// expectPrinted("`a${foo}b` + 'string'", "`a${foo}bstring`");
|
|
// expectPrinted("tag`template` + 'string'", 'tag`template` + "string"');
|
|
// expectPrinted("`template` + `a${foo}b`", "`templatea${foo}b`");
|
|
// expectPrinted("`a${foo}b` + `template`", "`a${foo}btemplate`");
|
|
// expectPrinted("`a${foo}b` + `x${bar}y`", "`a${foo}bx${bar}y`");
|
|
// expectPrinted(
|
|
// "`a${i}${j}bb` + `xxx${bar}yyyy`",
|
|
// "`a${i}${j}bbxxx${bar}yyyy`"
|
|
// );
|
|
// expectPrinted(
|
|
// "`a${foo}bb` + `xxx${i}${j}yyyy`",
|
|
// "`a${foo}bbxxx${i}${j}yyyy`"
|
|
// );
|
|
// expectPrinted(
|
|
// "`template` + tag`template2`",
|
|
// "`template` + tag`template2`"
|
|
// );
|
|
// expectPrinted(
|
|
// "tag`template` + `template2`",
|
|
// "tag`template` + `template2`"
|
|
// );
|
|
|
|
expectPrinted("123", "123");
|
|
expectPrinted("123 .toString()", "123 .toString()");
|
|
expectPrinted("-123", "-123");
|
|
expectPrinted("(-123).toString()", "(-123).toString()");
|
|
expectPrinted("-0", "-0");
|
|
expectPrinted("(-0).toString()", "(-0).toString()");
|
|
expectPrinted("-0 === 0", "!0");
|
|
|
|
expectPrinted("NaN", "NaN");
|
|
expectPrinted("NaN.toString()", "NaN.toString()");
|
|
expectPrinted("NaN === NaN", "!1");
|
|
|
|
expectPrinted("Infinity", "1 / 0");
|
|
expectPrinted("Infinity.toString()", "(1 / 0).toString()");
|
|
expectPrinted("(-Infinity).toString()", "(-1 / 0).toString()");
|
|
expectPrinted("Infinity === Infinity", "!0");
|
|
expectPrinted("Infinity === -Infinity", "!1");
|
|
|
|
expectPrinted("123n === 1_2_3n", "!0");
|
|
});
|
|
describe("type coercions", () => {
|
|
const dead = `
|
|
if ("") {
|
|
TEST_FAIL
|
|
}
|
|
|
|
if (false) {
|
|
TEST_FAIL
|
|
}
|
|
|
|
if (0) {
|
|
TEST_FAIL
|
|
}
|
|
|
|
if (void 0) {
|
|
TEST_FAIL
|
|
}
|
|
|
|
if (null) {
|
|
TEST_FAIL
|
|
}
|
|
|
|
var should_be_true = typeof "" === "string" || false
|
|
var should_be_false = typeof "" !== "string" && TEST_FAIL;
|
|
var should_be_false_2 = typeof true === "string" && TEST_FAIL;
|
|
var should_be_false_3 = typeof false === "string" && TEST_FAIL;
|
|
var should_be_false_4 = typeof 123n === "string" && TEST_FAIL;
|
|
var should_be_false_5 = typeof function(){} === "string" && TEST_FAIL;
|
|
var should_be_kept = typeof globalThis.BACON === "string" && TEST_OK;
|
|
var should_be_kept_1 = typeof TEST_OK === "string";
|
|
|
|
var should_be_kept_2 = TEST_OK ?? true;
|
|
var should_be_kept_4 = { "TEST_OK": true } ?? TEST_FAIL;
|
|
var should_be_false_6 = false ?? TEST_FAIL;
|
|
var should_be_true_7 = true ?? TEST_FAIL;
|
|
`;
|
|
const out = transpiler.transformSync(dead);
|
|
|
|
for (let line of out.split("\n")) {
|
|
it(line, () => {
|
|
if (line.includes("should_be_kept")) {
|
|
expect(line.includes("TEST_OK")).toBe(true);
|
|
}
|
|
|
|
if (line.includes("should_be_false")) {
|
|
if (!line.includes("= !1")) throw new Error(`Expected false in "${line}"`);
|
|
expect(line.includes("= !1")).toBe(true);
|
|
}
|
|
|
|
if (line.includes("TEST_FAIL")) {
|
|
throw new Error(`"${line}"\n\tshould not contain TEST_FAIL`);
|
|
}
|
|
});
|
|
}
|
|
});
|
|
});
|
|
|
|
it("raw template literal contents", () => {
|
|
expectPrinted("String.raw`\r`", "String.raw`\n`");
|
|
expectPrinted("String.raw`\r\n`", "String.raw`\n`");
|
|
expectPrinted("String.raw`\n`", "String.raw`\n`");
|
|
expectPrinted("String.raw`\r\r\r\r\r\n\r`", "String.raw`\n\n\n\n\n\n`");
|
|
expectPrinted("String.raw`\n\r`", "String.raw`\n\n`");
|
|
var code = `String.raw\`
|
|
<head>
|
|
<meta charset="UTF-8" />
|
|
<title>${"meow123"}</title>
|
|
<link rel="stylesheet" href="/css/style.css" />
|
|
</head>
|
|
\``;
|
|
var result = code;
|
|
expectPrinted(code, code);
|
|
|
|
code = `String.raw\`
|
|
<head>\r\n\n\r\r\r
|
|
<meta charset="UTF-8" />\r
|
|
<title>${"meow123"}</title>\n
|
|
|
|
\r
|
|
\r
|
|
\n\r
|
|
<link rel="stylesheet" href="/css/style.css" />
|
|
</head>
|
|
\``;
|
|
result = `String.raw\`
|
|
<head>\n\n\n\n
|
|
<meta charset="UTF-8" />
|
|
<title>${"meow123"}</title>\n
|
|
|
|
|
|
|
|
\n
|
|
<link rel="stylesheet" href="/css/style.css" />
|
|
</head>
|
|
\``;
|
|
expectPrinted(code, result);
|
|
});
|
|
|
|
describe("scan", () => {
|
|
it("reports all export names", () => {
|
|
const { imports, exports } = transpiler.scan(code);
|
|
|
|
expect(exports[0]).toBe("action");
|
|
expect(exports[2]).toBe("loader");
|
|
expect(exports[1]).toBe("default");
|
|
expect(exports).toHaveLength(3);
|
|
|
|
expect(imports.filter(({ path }) => path === "remix")).toHaveLength(1);
|
|
expect(imports.filter(({ path }) => path === "mod")).toHaveLength(0);
|
|
expect(imports.filter(({ path }) => path === "react")).toHaveLength(1);
|
|
expect(imports).toHaveLength(2);
|
|
});
|
|
});
|
|
|
|
describe("transform", () => {
|
|
// Async transform doesn't work in the test runner. Skipping for now.
|
|
// This might be caused by incorrectly using shared memory between the two files.
|
|
it.skip("supports macros", async () => {
|
|
const out = await transpiler.transform(`
|
|
import {keepSecondArgument} from 'macro:${require.resolve("./macro-check.js")}';
|
|
|
|
export default keepSecondArgument("Test failed", "Test passed");
|
|
export function otherNamesStillWork() {}
|
|
`);
|
|
expect(out.includes("Test failed")).toBe(false);
|
|
expect(out.includes("Test passed")).toBe(true);
|
|
|
|
// ensure both the import and the macro function call are removed
|
|
expect(out.includes("keepSecondArgument")).toBe(false);
|
|
expect(out.includes("otherNamesStillWork")).toBe(true);
|
|
});
|
|
|
|
it.todo("sync supports macros", () => {
|
|
const out = transpiler.transformSync(`
|
|
import {keepSecondArgument} from 'macro:${import.meta.dir}/macro-check.js';
|
|
|
|
export default keepSecondArgument("Test failed", "Test passed");
|
|
export function otherNamesStillWork() {
|
|
|
|
}
|
|
`);
|
|
expect(out.includes("Test failed")).toBe(false);
|
|
expect(out.includes("Test passed")).toBe(true);
|
|
|
|
expect(out.includes("keepSecondArgument")).toBe(false);
|
|
expect(out.includes("otherNamesStillWork")).toBe(true);
|
|
});
|
|
|
|
it("special identifier in import statement", () => {
|
|
const out = transpiler.transformSync(`
|
|
import {ɵtest} from 'foo'
|
|
`);
|
|
|
|
expect(out).toBe('import {ɵtest} from "foo";\n');
|
|
});
|
|
|
|
const importLines = ["import {createElement, bacon} from 'react';", "import {bacon, createElement} from 'react';"];
|
|
describe("sync supports macros remap", () => {
|
|
for (let importLine of importLines) {
|
|
it.todo(importLine, () => {
|
|
var thisCode = `
|
|
${importLine}
|
|
|
|
export default bacon("Test failed", "Test passed");
|
|
export function otherNamesStillWork() {
|
|
return createElement("div");
|
|
}
|
|
|
|
`;
|
|
var out = transpiler.transformSync(thisCode);
|
|
try {
|
|
expect(out.includes("Test failed")).toBe(false);
|
|
expect(out.includes("Test passed")).toBe(true);
|
|
|
|
expect(out.includes("bacon")).toBe(false);
|
|
expect(out.includes("createElement")).toBe(true);
|
|
} catch (e) {
|
|
console.log("Failing code:\n\n" + out + "\n");
|
|
throw e;
|
|
}
|
|
});
|
|
}
|
|
});
|
|
|
|
it.todo("macro remap removes import statement if its the only used one", () => {
|
|
const out = transpiler.transformSync(`
|
|
import {bacon} from 'react';
|
|
|
|
export default bacon("Test failed", "Test passed");
|
|
`);
|
|
|
|
expect(out.includes("Test failed")).toBe(false);
|
|
expect(out.includes("Test passed")).toBe(true);
|
|
|
|
expect(out.includes("bacon")).toBe(false);
|
|
expect(out.includes("import")).toBe(false);
|
|
});
|
|
|
|
it("removes types", () => {
|
|
expect(code.includes("mod")).toBe(true);
|
|
expect(code.includes("xx")).toBe(true);
|
|
expect(code.includes("ActionFunction")).toBe(true);
|
|
expect(code.includes("LoaderFunction")).toBe(true);
|
|
expect(code.includes("ReactNode")).toBe(true);
|
|
expect(code.includes("React")).toBe(true);
|
|
expect(code.includes("Component")).toBe(true);
|
|
const out = transpiler.transformSync(code);
|
|
|
|
expect(out.includes("ActionFunction")).toBe(false);
|
|
expect(out.includes("LoaderFunction")).toBe(false);
|
|
expect(out.includes("mod")).toBe(false);
|
|
expect(out.includes("xx")).toBe(false);
|
|
expect(out.includes("ReactNode")).toBe(false);
|
|
const { exports } = transpiler.scan(out);
|
|
exports.sort();
|
|
|
|
expect(exports[0]).toBe("action");
|
|
expect(exports[2]).toBe("loader");
|
|
expect(exports[1]).toBe("default");
|
|
expect(exports).toHaveLength(3);
|
|
});
|
|
});
|
|
|
|
describe("edge cases", () => {
|
|
it.todo("import statement with quoted specifier", () => {
|
|
expectPrinted_(`import { "x.y" as xy } from "bar";`, `import {"x.y" as xy} from "bar"`);
|
|
});
|
|
|
|
it('`str` + "``"', () => {
|
|
expectPrinted_('const x = `str` + "``";', "const x = `str\\`\\``");
|
|
expectPrinted_('const x = `` + "`";', "const x = `\\``");
|
|
expectPrinted_('const x = `` + "``";', "const x = `\\`\\``");
|
|
expectPrinted_('const x = "``" + ``;', "const x = `\\`\\``");
|
|
});
|
|
});
|
|
|
|
it("scan on empty file does not segfault", () => {
|
|
new Bun.Transpiler().scan("");
|
|
});
|
|
|
|
it("scanImports on empty file does not segfault", () => {
|
|
new Bun.Transpiler().scanImports("");
|
|
});
|
|
|
|
it("preserves exotic directives", () => {
|
|
expect(
|
|
new Bun.Transpiler().transformSync(`"use client";
|
|
console.log("boop");
|
|
`),
|
|
).toBe(
|
|
`"use client";
|
|
console.log("boop");
|
|
`,
|
|
);
|
|
});
|
|
it("does not preserve use strict (for now)", () => {
|
|
expect(
|
|
new Bun.Transpiler().transformSync(`"use strict";
|
|
console.log("boop");
|
|
`),
|
|
).toBe(
|
|
`console.log("boop");
|
|
`,
|
|
);
|
|
});
|
|
|
|
it("can parse 'a<b>' as typescript", () => {
|
|
ts.expectPrinted("a<b>", "a");
|
|
expect(new Bun.Transpiler({ loader: "ts" }).transformSync(`a<b>`)).toBe(`a;\n`);
|
|
});
|
|
|
|
const prepareForSnapshot = code => {
|
|
return code.replace(/(__using|__callDispose)_([a-z0-9]+)/g, "$1");
|
|
};
|
|
const expectCapturePrintedSnapshot = code => {
|
|
const result = parsed(`(async() => {${code}})()`, false, false);
|
|
expect(result).toEndWith("})();\n");
|
|
const of_relevance = result
|
|
.slice(result.indexOf("() => {") + 9, result.lastIndexOf("})();") - 1)
|
|
.trim()
|
|
.split("\n")
|
|
.map(x => x.trim())
|
|
.filter(x => x.length > 0)
|
|
.join("\n");
|
|
expect(prepareForSnapshot(of_relevance)).toMatchSnapshot();
|
|
};
|
|
const expectPrintedSnapshot = code => {
|
|
expect(prepareForSnapshot(parsed(`${code}`, false, false))).toMatchSnapshot();
|
|
};
|
|
|
|
it("using statements work right", () => {
|
|
expectCapturePrintedSnapshot(`using x = a;`);
|
|
expectCapturePrintedSnapshot(`await using x = a;`);
|
|
|
|
expectCapturePrintedSnapshot(`for (using a of b) c(a)`);
|
|
expectCapturePrintedSnapshot(`for await (using a of b) c(a)`);
|
|
expectCapturePrintedSnapshot(`for (await using a of b) c(a)`);
|
|
expectCapturePrintedSnapshot(`for await (await using a of b) c(a)`);
|
|
|
|
expectCapturePrintedSnapshot(`for (using a of b) { c(a); a(c) }`);
|
|
expectCapturePrintedSnapshot(`for await (using a of b) { c(a); a(c) }`);
|
|
expectCapturePrintedSnapshot(`for (await using a of b) { c(a); a(c) }`);
|
|
expectCapturePrintedSnapshot(`for await (await using a of b) { c(a); a(c) }`);
|
|
});
|
|
|
|
it("using top level", () => {
|
|
expectPrintedSnapshot(`
|
|
using a = b;
|
|
export function c(e) {
|
|
using f = g(a);
|
|
return f.h;
|
|
}
|
|
await using j = c(i);
|
|
using k = l(m);
|
|
export { k };
|
|
import { using } from 'n';
|
|
using o = using;
|
|
await using p = await using;
|
|
export var q = r;
|
|
`);
|
|
});
|
|
});
|
|
|
|
describe("await can only be used inside an async function message", () => {
|
|
var transpiler = new Bun.Transpiler({
|
|
logLevel: "debug",
|
|
});
|
|
|
|
function assertError(code, hasNote = false) {
|
|
try {
|
|
transpiler.transformSync(code);
|
|
expect.unreachable();
|
|
} catch (e) {
|
|
function handle(error) {
|
|
expect(error.message).toBe('"await" can only be used inside an "async" function');
|
|
|
|
if (hasNote) {
|
|
expect(error.notes).toHaveLength(1);
|
|
expect(error.notes[0].message).toBe('Consider adding the "async" keyword here');
|
|
expect(error.notes[0].position.lineText).toContain("foo");
|
|
} else {
|
|
expect(error.notes).toHaveLength(0);
|
|
}
|
|
}
|
|
if (e instanceof AggregateError) {
|
|
handle(e.errors[0]);
|
|
} else {
|
|
expect.unreachable();
|
|
}
|
|
}
|
|
}
|
|
it("in object method", () => {
|
|
assertError(
|
|
`const x = {
|
|
foo() {
|
|
await bar();
|
|
}
|
|
}`,
|
|
true,
|
|
);
|
|
});
|
|
|
|
it("in class method", () => {
|
|
assertError(
|
|
`class X {
|
|
foo() {
|
|
await bar();
|
|
}
|
|
}`,
|
|
true,
|
|
);
|
|
});
|
|
|
|
it("in function statement", () => {
|
|
assertError(
|
|
`function foo() {
|
|
await bar();
|
|
}`,
|
|
true,
|
|
);
|
|
});
|
|
|
|
it("in function expression", () => {
|
|
assertError(
|
|
`const foo = function() {
|
|
await bar();
|
|
}`,
|
|
true,
|
|
);
|
|
});
|
|
|
|
it("in arrow function", () => {
|
|
assertError(
|
|
`const foo = () => {
|
|
await bar();
|
|
}`,
|
|
false,
|
|
);
|
|
});
|
|
|
|
it("in arrow function with block body", () => {
|
|
assertError(
|
|
`const foo = () => {
|
|
await bar();
|
|
}`,
|
|
false,
|
|
);
|
|
});
|
|
|
|
it("in arrow function with expression body", () => {
|
|
assertError(`const foo = () => await bar();`, false);
|
|
});
|
|
});
|