use crate::tester::{test, test_minify, test_minify_same, test_without_source}; #[test] fn module_decl() { test("export * as foo from 'foo'", "export * as foo from \"foo\";\n"); test("import x from './foo.js' with {}", "import x from \"./foo.js\" with {};\n"); test("import {} from './foo.js' with {}", "import {} from \"./foo.js\" with {};\n"); test("export * from './foo.js' with {}", "export * from \"./foo.js\" with {};\n"); } #[test] fn expr() { test("new (foo()).bar();", "new (foo()).bar();\n"); test_minify("x in new Error()", "x in new Error();"); test("1000000000000000128.0.toFixed(0)", "0xde0b6b3a7640080.toFixed(0);\n"); test_minify("1000000000000000128.0.toFixed(0)", "0xde0b6b3a7640080.toFixed(0);"); test_minify("throw 'foo'", "throw\"foo\";"); test_minify("return 'foo'", "return\"foo\";"); test_minify("return class {}", "return class{};"); test_minify("return async function foo() {}", "return async function foo(){};"); test_minify_same("return super();"); test_minify_same("return new.target;"); test_minify_same("throw await 1;"); test_minify_same("await import(\"\");"); } #[test] fn private_in() { test( "class Foo { #test; bar() { if (!(#test in Foo)) { } } }", "class Foo {\n\t#test;\n\tbar() {\n\t\tif (!(#test in Foo)) {}\n\t}\n}\n", ); test( "class Foo { #test; bar() { #field in {} << 0 } }", "class Foo {\n\t#test;\n\tbar() {\n\t\t#field in {} << 0;\n\t}\n}\n", ); } #[test] fn access_property() { test( "export default class Foo { @x @y accessor #aDef = 1 }", "export default class Foo {\n\t@x @y accessor #aDef = 1;\n}\n", ); } #[test] fn for_stmt() { test("for (let x = 0; x < 10; x++) {}", "for (let x = 0; x < 10; x++) {}\n"); test("for (;;) {}", "for (;;) {}\n"); test("for (let x = 1;;) {}", "for (let x = 1;;) {}\n"); test("for (;true;) {}", "for (; true;) {}\n"); test("for (;;i++) {}", "for (;; i++) {}\n"); test("for (using x = 1;;) {}", "for (using x = 1;;) {}\n"); // TODO // test( // "for (var a = 1 || (2 in {}) in { x: 1 }) count++;", // "for (var a = 1 || (2 in {}) in {x: 1}) count++;\n", // ); } #[test] fn shorthand() { test("let _ = { x }", "let _ = { x };\n"); test("let { x } = y", "let { x } = y;\n"); test("({ x: (x) })", "({ x });\n"); test("({ x } = y)", "({x} = y);\n"); // https://github.com/tc39/test262/blob/05c45a4c430ab6fee3e0c7f0d47d8a30d8876a6d/test/language/expressions/object/__proto__-permitted-dup-shorthand.js test("var obj = { __proto__, __proto__, };", "var obj = {\n\t__proto__,\n\t__proto__\n};\n"); test("var obj = { __proto__: __proto__, };", "var obj = { __proto__: __proto__ };\n"); } #[test] fn unicode_escape() { test("console.log('你好');", "console.log(\"你好\");\n"); test("console.log('こんにちは');", "console.log(\"こんにちは\");\n"); test("console.log('안녕하세요');", "console.log(\"안녕하세요\");\n"); test("console.log('🧑‍🤝‍🧑');", "console.log(\"🧑‍🤝‍🧑\");\n"); } #[test] fn regex() { fn test_all(source: &str, expect: &str, minify: &str) { test(source, expect); test_minify(source, minify); test_without_source(source, expect); } test_all("/regex/giv", "/regex/giv;\n", "/regex/giv;"); test_all( r"/(.)(.)(.)(.)(.)(.)(.)(.)\8\8/", "/(.)(.)(.)(.)(.)(.)(.)(.)\\8\\8/;\n", "/(.)(.)(.)(.)(.)(.)(.)(.)\\8\\8/;", ); test_all( r"/\n\cM\0\x41\u{1f600}\./u", "/\\n\\cM\\0\\x41\\u{1f600}\\./u;\n", "/\\n\\cM\\0\\x41\\u{1f600}\\./u;", ); test_all(r"/\n\cM\0\x41\./u", "/\\n\\cM\\0\\x41\\./u;\n", "/\\n\\cM\\0\\x41\\./u;"); test_all( r"/\n\cM\0\x41\u1234\./", "/\\n\\cM\\0\\x41\\u1234\\./;\n", "/\\n\\cM\\0\\x41\\u1234\\./;", ); } #[test] fn comma() { test("[1, 2, 3]", "[\n\t1,\n\t2,\n\t3\n];\n"); test("[1, 2, 3,]", "[\n\t1,\n\t2,\n\t3\n];\n"); test("[,]", "[,];\n"); test("[,,]", "[, ,];\n"); test("[,1]", "[, 1];\n"); test_minify("1, 2, 3", "1,2,3;"); test_minify("1, a = b, 3", "1,a=b,3;"); test_minify("1, (2, 3), 4", "1,2,3,4;"); } #[test] fn assignment() { test("(let[0] = 100);", "(let)[0] = 100;\n"); test_minify("a = b ? c : d", "a=b?c:d;"); test_minify("[a,b] = (1, 2)", "[a,b]=(1,2);"); // `{a,b}` is a block, must wrap the whole expression to be an assignment expression test_minify("({a,b} = (1, 2))", "({a,b}=(1,2));"); test_minify("a *= yield b", "a*=yield b;"); test_minify("a /= () => {}", "a/=()=>{};"); test_minify("a %= async () => {}", "a%=async()=>{};"); test_minify("a -= (1, 2)", "a-=(1,2);"); test_minify("a >>= b >>= c", "a>>=b>>=c;"); } #[test] fn r#yield() { test_minify("function *foo() { yield }", "function*foo(){yield}"); test_minify("function *foo() { yield * a ? b : c }", "function*foo(){yield*a?b:c}"); test_minify("function *foo() { yield * yield * a }", "function*foo(){yield*yield*a}"); test_minify("function *foo() { yield * () => {} }", "function*foo(){yield*()=>{}}"); test_minify("function *foo() { yield * async () => {} }", "function*foo(){yield*async()=>{}}"); test_minify("function *foo() { yield a ? b : c }", "function*foo(){yield a?b:c}"); test_minify("function *foo() { yield yield a }", "function*foo(){yield yield a}"); test_minify("function *foo() { yield () => {} }", "function*foo(){yield ()=>{}}"); test_minify("function *foo() { yield async () => {} }", "function*foo(){yield async()=>{}}"); test_minify( "function *foo() { yield { a } = [ b ] = c ? b : d }", "function*foo(){yield {a}=[b]=c?b:d}", ); // TODO: remove the extra space in `yield (a,b)` test_minify("function *foo() { yield (a, b) }", "function*foo(){yield (a,b)}"); test_minify("function *foo() { yield a, b }", "function*foo(){yield a,b}"); } #[test] fn arrow() { test_minify("x => a, b", "(x)=>a,b;"); test_minify("x => (a, b)", "(x)=>(a,b);"); test_minify("x => (a => b)", "(x)=>(a)=>b;"); test_minify("x => y => a, b", "(x)=>(y)=>a,b;"); test_minify("x => y => (a = b)", "(x)=>(y)=>a=b;"); test_minify("x => y => z => a = b, c", "(x)=>(y)=>(z)=>a=b,c;"); test_minify("x => y => z => a = (b, c)", "(x)=>(y)=>(z)=>a=(b,c);"); test_minify("x => ({} + 0)", "(x)=>({})+0;"); } #[test] fn conditional() { test_minify("a ? b : c", "a?b:c;"); test_minify("a ? (b, c) : (d, e)", "a?(b,c):(d,e);"); test_minify("a ? b : c ? b : c", "a?b:c?b:c;"); test_minify("(a ? b : c) ? b : c", "(a?b:c)?b:c;"); test_minify("a, b ? c : d", "a,b?c:d;"); test_minify("(a, b) ? c : d", "(a,b)?c:d;"); test_minify("a = b ? c : d", "a=b?c:d;"); test_minify("(a = b) ? c : d", "(a=b)?c:d;"); } #[test] fn coalesce() { test_minify("a ?? b", "a??b;"); test_minify("a ?? b ?? c ?? d", "a??b??c??d;"); test_minify("a ?? (b ?? (c ?? d))", "a??b??c??d;"); test_minify("(a ?? (b ?? (c ?? d)))", "a??b??c??d;"); test_minify("a, b ?? c", "a,b??c;"); test_minify("(a, b) ?? c", "(a,b)??c;"); test_minify("a, b ?? c, d", "a,b??c,d;"); test_minify("a, b ?? (c, d)", "a,b??(c,d);"); test_minify("a = b ?? c", "a=b??c;"); test_minify("a ?? (b = c)", "a??(b=c);"); test_minify("(a | b) ?? (c | d)", "a|b??c|d;"); } #[test] fn logical_or() { test_minify("a || b || c", "a||b||c;"); test_minify("(a || (b || c)) || d", "a||b||c||d;"); test_minify("a || (b || (c || d))", "a||b||c||d;"); test_minify("a || b && c", "a||b&&c;"); test_minify("(a || b) && c", "(a||b)&&c;"); test_minify("a, b || c, d", "a,b||c,d;"); test_minify("(a, b) || (c, d)", "(a,b)||(c,d);"); test_minify("(a && b) || (c && d)", "a&&b||c&&d;"); test_minify("a && b || c && d", "a&&b||c&&d;"); } #[test] fn logical_and() { test_minify("a && b && c", "a&&b&&c;"); test_minify("a && ((b && c) && d)", "a&&b&&c&&d;"); test_minify("((a && b) && c) && d", "a&&b&&c&&d;"); test_minify("(a || b) && (c || d)", "(a||b)&&(c||d);"); test_minify("a, b && c, d", "a,b&&c,d;"); test_minify("(a, b) && (c, d)", "(a,b)&&(c,d);"); test_minify("a || b && c || d", "a||b&&c||d;"); } #[test] fn bitwise_or() { test_minify("a | b | c", "a|b|c;"); test_minify("(a | b) | c", "a|b|c;"); test_minify("a | (b | c)", "a|(b|c);"); test_minify("a | b ^ c", "a|b^c;"); test_minify("a | (b ^ c)", "a|b^c;"); test_minify("a | (b && c)", "a|(b&&c);"); test_minify("a | b && c", "a|b&&c;"); test_minify("(a ^ b) | (c ^ d)", "a^b|c^d;"); test_minify("(a, b) | (c, d)", "(a,b)|(c,d);"); test_minify("a, b | c, d", "a,b|c,d;"); } #[test] fn bitwise_xor() { test_minify("a ^ b ^ c", "a^b^c;"); test_minify("(a ^ b) ^ c", "a^b^c;"); test_minify("a ^ (b ^ c)", "a^(b^c);"); test_minify("a | b & c", "a|b&c;"); test_minify("a | (b & c)", "a|b&c;"); test_minify("a | (b || c)", "a|(b||c);"); test_minify("a | b || c", "a|b||c;"); test_minify("(a, b) ^ (c, d)", "(a,b)^(c,d);"); test_minify("(a | b) ^ (c | d)", "(a|b)^(c|d);"); test_minify("a, b ^ c, d", "a,b^c,d;"); } #[test] fn bitwise_and() { test_minify("a & b & c", "a&b&c;"); test_minify("((a & b) & c) & d", "a&b&c&d;"); test_minify("a & (b & (c & d))", "a&(b&(c&d));"); test_minify("a & b == c", "a&b==c;"); test_minify("a & (b == c)", "a&b==c;"); test_minify("a == b & c", "a==b&c;"); test_minify("(a == b) & c", "a==b&c;"); test_minify("a ^ b & c", "a^b&c;"); test_minify("(a ^ b) & c", "(a^b)&c;"); test_minify("(a, b) & (c, d)", "(a,b)&(c,d);"); test_minify("a, b & c, d", "a,b&c,d;"); } #[test] fn equality() { test_minify("a == b != c === d !== e", "a==b!=c===d!==e;"); test_minify("a == (b != (c === (d !== e)))", "a==(b!=(c===(d!==e)));"); test_minify("(((a == b) != c) === d) !== e", "a==b!=c===d!==e;"); test_minify("a > b == c < d", "a>b==c b) == (c < d)", "a>b==c {\nself.postMessage(module.default + module1.msg1 + import.meta.env.BASE_URL)})", "import(\n\t/* @vite-ignore */\n\tmodule1Url\n).then((module1) => {\n\tself.postMessage(module.default + module1.msg1 + import.meta.env.BASE_URL);\n});\n", ); } // followup from https://github.com/oxc-project/oxc/pull/6422 #[test] fn in_expr_in_sequence_in_for_loop_init() { test( "for (l = ('foo' in bar), i; i < 10; i += 1) {}", "for (l = (\"foo\" in bar), i; i < 10; i += 1) {}\n", ); test( "for (('hidden' in a) && (m = a.hidden), r = 0; s > r; r++) {}", "for ((\"hidden\" in a) && (m = a.hidden), r = 0; s > r; r++) {}\n", ); } #[test] fn in_expr_in_arrow_function_expression() { test("() => ('foo' in bar)", "() => \"foo\" in bar;\n"); test("() => 'foo' in bar", "() => \"foo\" in bar;\n"); test("() => { ('foo' in bar) }", "() => {\n\t\"foo\" in bar;\n};\n"); } #[test] fn big_int() { test("9007199254740991n;", "9007199254740991n;\n"); test("-9007199254740991n;", "-9007199254740991n;\n"); test("-90_0719_92547_40991n;", "-9007199254740991n;\n"); test("+9007199254740991n;", "+9007199254740991n;\n"); test("1000n", "1000n;\n"); test("-15n", "-15n;\n"); test("100_000_000n;", "100000000n;\n"); test("10000000000000000n;", "10000000000000000n;\n"); test("0n;", "0n;\n"); test("+0n;", "+0n;\n"); test("-0n;", "-0n;\n"); test("0x1_0n;", "0x10n;\n"); test("0x10n;", "0x10n;\n"); test("0b1_01n;", "0b101n;\n"); test("0b101n;", "0b101n;\n"); test("0b101_101n;", "0b101101n;\n"); test("0b10_1n", "0b101n;\n"); test("0o13n;", "0o13n;\n"); test("0o7n", "0o7n;\n"); test("0x2_0n", "0x20n;\n"); test("0xfabn", "0xfabn;\n"); test("0xaef_en;", "0xaefen;\n"); test("0xaefen;", "0xaefen;\n"); } #[test] #[ignore = "Minify bigint is not implemented."] fn big_int_minify() { test_minify("9007199254740991n", "9007199254740991n;"); test_minify("-9007199254740991n;", "-9007199254740991n;"); test_minify("-90_0719_92547_40991n;", "-9007199254740991n;"); test_minify("+9007199254740991n;", "+9007199254740991n;"); test_minify("1000n", "1000n;"); test_minify("-15n", "-15n;"); test_minify("100_000_000n;", "100000000n;"); test_minify("10000000000000000n;", "0x2386f26fc10000n;"); test_minify("0n;", "0n;"); test_minify("+0n;", "+0n;"); test_minify("-0n;", "-0n;"); test_minify("0x1_0n;", "16n;"); test_minify("0x10n;", "16n;"); test_minify("0b1_01n;", "5n;"); test_minify("0b101n;", "5n;"); test_minify("0b101_101n;", "45n;"); test_minify("0b10_1n", "5n;"); test_minify("0o13n;", "11n;"); test_minify("0o7n", "7n;"); test_minify("0x2_0n", "32n;"); test_minify("0xfabn", "4011n;"); test_minify("0xaef_en;", "44798n;"); test_minify("0xaefen;", "44798n;"); }