import { expect, it, describe } from "bun:test"; describe("Bun.Transpiler", () => { 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", }); 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 = err.errors[0]; } expect(er.message).toBe(message); return; } throw new Error("Expected parse error for code\n\t" + code); }, }; it("normalizes \\r\\n", () => { ts.expectPrinted_( "console.log(`\r\n\r\n\r\n`)", "console.log(`\n\n\n`);\n", ); }); describe("TypeScript", () => { it("import Foo = Baz.Bar", () => { ts.expectPrinted_( "import Foo = Baz.Bar;\nexport default Foo;", "const Foo = Baz.Bar;\nexport default Foo", ); }); 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("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 = { f(s) { }, g(s) { } } satisfies { g(s: string): void } & Record;", "let obj = { f(s) {\n}, g(s) {\n} };\n", ); ts.expectPrinted_( "const car = { start() { }, move(d) { }, stop() { } } satisfies Movable & Record;", "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;", "const a = { x: 10 };\n", ); ts.expectPrinted_( 'const p = { a: 0, b: "hello", x: 8 } satisfies Partial>;', 'const p = { a: 0, b: "hello", x: 8 };\n', ); ts.expectPrinted_( 'const p = { a: 0, b: "hello", x: 8 } satisfies Record;', '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;", "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
{user.name}
; } `); }); 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("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`, }, }, }); 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("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
hi
`); 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("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
hi
`); 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("JSX", () => { var bun = new Bun.Transpiler({ loader: "jsx", define: { "process.env.NODE_ENV": JSON.stringify("development"), }, }); expect(bun.transformSync("export var foo =
")).toBe( `export var foo = jsx("div", { foo: true }, undefined, false, undefined, this); `, ); expect(bun.transformSync("export var foo =
")).toBe( `export var foo = jsx("div", { foo }, undefined, false, undefined, this); `, ); expect(bun.transformSync("export var foo =
")).toBe( `export var foo = jsx("div", { ...foo }, undefined, false, undefined, this); `, ); expect(bun.transformSync("export var hi =
")).toBe( `export var hi = jsx("div", { foo }, undefined, false, undefined, this); `, ); expect(bun.transformSync("export var hi =
")).toBe( `export var hi = jsx("div", { baz: foo.bar.baz }, undefined, false, undefined, this); `, ); expect(bun.transformSync("export var hi =
")).toBe( `export var hi = jsx("div", { baz: foo?.bar?.baz }, undefined, false, undefined, this); `, ); expect( bun.transformSync("export var hi =
"), ).toBe( `export var hi = jsx("div", { baz: foo["baz"].bar?.baz }, undefined, false, undefined, this); `, ); // cursed expect( bun.transformSync( "export var hi =
true}.name].hi} />", ), ).toBe( `export var hi = jsx("div", { hi: foo[{ name: () => true }.name].hi }, undefined, false, undefined, this); `, ); expect( bun.transformSync("export var hi = "), ).toBe( `export var hi = jsx(Foo, { NODE_ENV: "development" }, undefined, false, undefined, this); `, ); expect( bun.transformSync("export var hi =
"), ).toBe( `export var hi = jsx("div", { baz: foo["baz"].bar?.baz }, undefined, false, undefined, this); `, ); try { bun.transformSync("export var hi =
"); throw new Error("Expected error"); } catch (e) { expect(e.errors[0].message.includes('Expected ">"')).toBe(true); } expect( bun.transformSync("export var hi =
"), ).toBe( `export var hi = jsx("div", { Foo, children: jsx(Foo, {}, undefined, false, undefined, this) }, undefined, false, undefined, this); `, ); expect( bun.transformSync("export var hi =
"), ).toBe( `export var hi = jsx("div", { Foo, children: jsx(Foo, {}, undefined, false, undefined, this) }, undefined, false, undefined, this); `, ); expect(bun.transformSync("export var hi =
{123}}
").trim()).toBe( `export var hi = jsx("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, }); it("inlines static JSX into object literals", () => { expect( inliner .transformSync( ` export var hi =
{123}
export var hiWithKey =
{123}
export var hiWithRef =
{123}
export var ComponentThatChecksDefaultProps = export var ComponentThatChecksDefaultPropsAndHasChildren = my child export var ComponentThatHasSpreadCausesDeopt = `.trim(), ) .trim(), ).toBe( `var $$typeof = Symbol.for("react.element"); export var hi = { $$typeof, type: "div", key: null, ref: null, props: { children: 123 }, _owner: null }; export var hiWithKey = { $$typeof, type: "div", key: "hey", ref: null, props: { children: 123 }, _owner: null }; export var hiWithRef = jsx("div", { ref: foo, children: 123 }); export var ComponentThatChecksDefaultProps = { $$typeof, type: Hello, key: null, ref: null, props: Hello.defaultProps || {}, _owner: null }; export var ComponentThatChecksDefaultPropsAndHasChildren = { $$typeof, type: Hello, key: null, ref: null, props: __merge({ children: "my child" }, Hello.defaultProps), _owner: null }; export var ComponentThatHasSpreadCausesDeopt = jsx(Hello, { ...spread }); `.trim(), ); }); }); 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 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_( `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("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("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("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("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("rewrite string to length", () => { expectPrinted_( `export const foo = "a".length + "b".length;`, `export const foo = 1 + 1`, ); expectBunPrinted_( `export const foo = "a".length + "b".length;`, `export const foo = 2`, ); }); describe("Bun.js", () => { it("require -> import.meta.require", () => { expectBunPrinted_( `export const foo = require('bar.node')`, `export const foo = import.meta.require("bar.node")`, ); }); it("require.resolve -> import.meta.resolveSync", () => { expectBunPrinted_( `export const foo = require.resolve('bar.node')`, `export const foo = import.meta.resolveSync("bar.node")`, ); }); it('require.resolve(path, {paths: ["blah"]}) -> import.meta.resolveSync', () => { expectBunPrinted_( `export const foo = require.resolve('bar.node', {paths: ["blah"]})`, `export const foo = import.meta.resolveSync("bar.node", { paths: ["blah"] })`, ); }); }); describe("Browsers", () => { it('require.resolve("my-module") -> "/resolved/my-module"', () => { // the module resolver & linker doesn't run with Bun.Transpiler // so in this test, it becomes the same path string expectPrinted_( `export const foo = require.resolve('my-module')`, `export const foo = "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`); }); it("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' ); expectPrinted_("for (x in y) ;", "for (x in y) {\n}"); expectPrinted_("for ([] in y) ;", "for ([] in y) {\n}"); expectPrinted_("for ({} in y) ;", "for ({} in y) {\n}"); expectPrinted_("for ((x) in y) ;", "for (x in y) {\n}"); expectParseError("for (([]) in y) ;", "Invalid assignment target"); expectParseError("for (({}) in y) ;", "Invalid assignment target"); expectPrinted_("for (x of y) ;", "for (x of y) {\n}"); expectPrinted_("for ([] of y) ;", "for ([] of y) {\n}"); expectPrinted_("for ({} of y) ;", "for ({} of y) {\n}"); expectPrinted_("for ((x) of y) ;", "for (x of y) {\n}"); 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 "="'); expectPrinted_( "for ([{a = {}}] in b) {}", "for ([{ a = {} }] in b) {\n}", ); expectPrinted_( "for ([{a = {}}] of b) {}", "for ([{ a = {} }] of b) {\n}", ); expectPrinted_("for ({a = {}} in b) {}", "for ({ a = {} } in b) {\n}"); expectPrinted_("for ({a = {}} of b) {}", "for ({ a = {} } of b) {\n}"); 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"'); }); }); 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", () => { it("unary operator", () => { expectPrinted("a = !(b, c)", "a = (b , !c)"); }); it("substitution", () => { var transpiler = new Bun.Transpiler({ inline: true, platform: "bun", allowBunRuntime: false, }); 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 false;"); 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 check("let x = 1; x = 2", "let x = 1;\nx = 2;"); check("let x = 1; x += 2", "let x = 1;\nx += 2;"); check("let x = 1; x ||= 2", "let x = 1;\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", () => { 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", "false"); 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"', "true"); expectPrinted("typeof {foo: 123} === typeof {bar: 123}", "true"); expectPrinted("typeof {foo: 123} !== typeof 123", "true"); expectPrinted("undefined === undefined", "true"); expectPrinted("undefined !== undefined", "false"); expectPrinted("undefined == undefined", "true"); expectPrinted("undefined != undefined", "false"); expectPrinted("null === null", "true"); expectPrinted("null !== null", "false"); expectPrinted("null == null", "true"); expectPrinted("null != null", "false"); expectPrinted("undefined === null", "undefined === null"); expectPrinted("undefined !== null", "undefined !== null"); expectPrinted("undefined == null", "undefined == null"); expectPrinted("undefined != null", "undefined != null"); expectPrinted("true === true", "true"); expectPrinted("true === false", "false"); expectPrinted("true !== true", "false"); expectPrinted("true !== false", "true"); expectPrinted("true == true", "true"); expectPrinted("true == false", "false"); expectPrinted("true != true", "false"); expectPrinted("true != false", "true"); expectPrinted("1 === 1", "true"); expectPrinted("1 === 2", "false"); expectPrinted("1 === '1'", '1 === "1"'); expectPrinted("1 == 1", "true"); expectPrinted("1 == 2", "false"); expectPrinted("1 == '1'", '1 == "1"'); expectPrinted("1 !== 1", "false"); expectPrinted("1 !== 2", "true"); expectPrinted("1 !== '1'", '1 !== "1"'); expectPrinted("1 != 1", "false"); expectPrinted("1 != 2", "true"); expectPrinted("1 != '1'", '1 != "1"'); expectPrinted("'a' === '\\x61'", "true"); expectPrinted("'a' === '\\x62'", "false"); expectPrinted("'a' === 'abc'", "false"); expectPrinted("'a' !== '\\x61'", "false"); expectPrinted("'a' !== '\\x62'", "true"); expectPrinted("'a' !== 'abc'", "true"); expectPrinted("'a' == '\\x61'", "true"); expectPrinted("'a' == '\\x62'", "false"); expectPrinted("'a' == 'abc'", "false"); expectPrinted("'a' != '\\x61'", "false"); expectPrinted("'a' != '\\x62'", "true"); expectPrinted("'a' != 'abc'", "true"); 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"'); 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", "true"); expectPrinted("NaN", "NaN"); expectPrinted("NaN.toString()", "NaN.toString()"); expectPrinted("NaN === NaN", "false"); expectPrinted("Infinity", "Infinity"); expectPrinted("Infinity.toString()", "Infinity.toString()"); expectPrinted("(-Infinity).toString()", "(-Infinity).toString()"); expectPrinted("Infinity === Infinity", "true"); expectPrinted("Infinity === -Infinity", "false"); expectPrinted("123n === 1_2_3n", "true"); }); 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("= false")) throw new Error(`Expected false in "${line}"`); expect(line.includes("= false")).toBe(true); } if (line.includes("TEST_FAIL")) { throw new Error(`"${line}"\n\tshould not contain TEST_FAIL`); } }); } }); }); 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", () => { it("supports macros", async () => { const out = await transpiler.transform(` 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); // ensure both the import and the macro function call are removed expect(out.includes("keepSecondArgument")).toBe(false); expect(out.includes("otherNamesStillWork")).toBe(true); }); it("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(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("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); }); }); });