Compare commits
224 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
85786889bd | ||
|
|
4b88dfb8d9 | ||
|
|
c3aef23614 | ||
|
|
db94d21980 | ||
|
|
9634a9d1fd | ||
|
|
befb99bd71 | ||
|
|
02eb8baa1c | ||
|
|
c09f63aefb | ||
|
|
fdcc6d3a9c | ||
|
|
4fe2cac35e | ||
|
|
e219a9a78a | ||
|
|
c80eabd61e | ||
|
|
9b82f9be91 | ||
|
|
657d525c80 | ||
|
|
6a3fe9d1df | ||
|
|
ec7fadcb5b | ||
|
|
65adeba55d | ||
|
|
7fac839c62 | ||
|
|
8926a2f327 | ||
|
|
85968dee54 | ||
|
|
a7e7865e6b | ||
|
|
ef5f7fc25e | ||
|
|
902997b73d | ||
|
|
f18804fa06 | ||
|
|
8e2dff632e | ||
|
|
92c3fddd7a | ||
|
|
0d350b78bf | ||
|
|
1ad830facb | ||
|
|
64ebf6efe9 | ||
|
|
08391b8e1c | ||
|
|
d147d5d7f0 | ||
|
|
aae1fcd12d | ||
|
|
f67dd31cbb | ||
|
|
450aabaaa0 | ||
|
|
ea7829daf5 | ||
|
|
6577d641ac | ||
|
|
2340feff87 | ||
|
|
0c48b84540 | ||
|
|
1fefe3f1d1 | ||
|
|
0668fad5e9 | ||
|
|
d0e3f6955d | ||
|
|
6961c57d1e | ||
|
|
f5dbb672b9 | ||
|
|
f4ae267920 | ||
|
|
972b9f0bef | ||
|
|
668f96623c | ||
|
|
1b745494ce | ||
|
|
611abff49f | ||
|
|
4ba8b66c5a | ||
|
|
798121c9f3 | ||
|
|
b23b333d9d | ||
|
|
7621527a5f | ||
|
|
1a064b6e74 | ||
|
|
82772ccb12 | ||
|
|
7cbcd11440 | ||
|
|
980dbde171 | ||
|
|
8b05677c15 | ||
|
|
95090dbf24 | ||
|
|
7c5b6f349e | ||
|
|
e9c902b044 | ||
|
|
111366fca0 | ||
|
|
e368d39715 | ||
|
|
fd8dec61ad | ||
|
|
7880568d15 | ||
|
|
21fc8f4630 | ||
|
|
bf76e35772 | ||
|
|
ac1262dc97 | ||
|
|
498ac83541 | ||
|
|
6fc7a2ab6a | ||
|
|
f8b2215145 | ||
|
|
70ceda5398 | ||
|
|
ce75477670 | ||
|
|
e70b84895c | ||
|
|
8dbf0b042e | ||
|
|
83f7887e5d | ||
|
|
dff7b48921 | ||
|
|
f4f0d2a2dd | ||
|
|
06e3dbc089 | ||
|
|
55a230daa8 | ||
|
|
23b9f36bd8 | ||
|
|
7e88d52fae | ||
|
|
b9d5bba5fb | ||
|
|
8d23496e0f | ||
|
|
260431f4e0 | ||
|
|
7fa1dea9d0 | ||
|
|
d40631fd44 | ||
|
|
d320a6cde2 | ||
|
|
8cd95dd263 | ||
|
|
749a828fc5 | ||
|
|
ab42a90edb | ||
|
|
362abe0ffb | ||
|
|
e3798d9a76 | ||
|
|
c8d10b7cde | ||
|
|
7caab39e26 | ||
|
|
eff45eac0e | ||
|
|
1e787c556b | ||
|
|
df47632ecc | ||
|
|
eb08fed120 | ||
|
|
8b0c836515 | ||
|
|
5d4e6e3bdc | ||
|
|
d2a45ba441 | ||
|
|
de376c3d33 | ||
|
|
4a19575e74 | ||
|
|
e0695ef549 | ||
|
|
d6152e6a76 | ||
|
|
d930c705f6 | ||
|
|
254937754c | ||
|
|
ae4dbcb5b9 | ||
|
|
e13615549e | ||
|
|
a7698f8845 | ||
|
|
bbed9b13b1 | ||
|
|
2cff7c94e8 | ||
|
|
7576048118 | ||
|
|
3c1898fd65 | ||
|
|
e04429350f | ||
|
|
60f3b55156 | ||
|
|
689f8f504d | ||
|
|
ae51f76ba7 | ||
|
|
7eef86ed05 | ||
|
|
b1cfa71131 | ||
|
|
7b8570f16c | ||
|
|
bb225367cb | ||
|
|
bbca9de9cd | ||
|
|
ee9ceb79ca | ||
|
|
ac1f7d689b | ||
|
|
19d232badb | ||
|
|
d464be3f3f | ||
|
|
3094eaaa89 | ||
|
|
ce3c35fa8b | ||
|
|
5d9224deb8 | ||
|
|
45b6d23d36 | ||
|
|
f0de9a8b5d | ||
|
|
203f4b7ad9 | ||
|
|
4114431eec | ||
|
|
fb03561799 | ||
|
|
6ab26eef6c | ||
|
|
53b57ee57e | ||
|
|
16411dcb87 | ||
|
|
8bbfaacdae | ||
|
|
df980db4a8 | ||
|
|
8e4a19ffec | ||
|
|
d833e66d23 | ||
|
|
8aa650bcf9 | ||
|
|
d576495e5a | ||
|
|
a06e20304b | ||
|
|
4cccc01f3e | ||
|
|
97bd56b7e8 | ||
|
|
acf951a5bc | ||
|
|
324587f769 | ||
|
|
80efaa2f33 | ||
|
|
a1a212f639 | ||
|
|
a2b1b96752 | ||
|
|
c296a63fb3 | ||
|
|
10dd9d4eaf | ||
|
|
9b8deff64d | ||
|
|
f46209b7e5 | ||
|
|
bddb5a0102 | ||
|
|
3c161a6662 | ||
|
|
c58e174647 | ||
|
|
c53af3dfb1 | ||
|
|
f1f4a4dd82 | ||
|
|
ca49f6f41a | ||
|
|
8a82822654 | ||
|
|
ebe4e1ad28 | ||
|
|
a37ca558dd | ||
|
|
73a564343b | ||
|
|
4870747306 | ||
|
|
b179a2459f | ||
|
|
231c3d7c84 | ||
|
|
aed758ed5c | ||
|
|
0df028187d | ||
|
|
10fbf8e295 | ||
|
|
cf38b52afa | ||
|
|
e755d01a0b | ||
|
|
4084948d3b | ||
|
|
cea1fb5c58 | ||
|
|
1947a21824 | ||
|
|
6335b5fd8a | ||
|
|
daa8319b8a | ||
|
|
d5599604e8 | ||
|
|
072933f1d5 | ||
|
|
39df3a1680 | ||
|
|
03c5ecb2e3 | ||
|
|
40ef074cb3 | ||
|
|
78e3936cd4 | ||
|
|
e7be38b42a | ||
|
|
44394e61c9 | ||
|
|
f9055df44d | ||
|
|
51bdb7281b | ||
|
|
9c01511f84 | ||
|
|
9faee3b66a | ||
|
|
8ea1ced389 | ||
|
|
24619daf68 | ||
|
|
b89cc84c3a | ||
|
|
3016a78d85 | ||
|
|
2508481e33 | ||
|
|
48c46fa9a7 | ||
|
|
7da49b5709 | ||
|
|
d837a46ebd | ||
|
|
d4303b62cc | ||
|
|
7d595e2eac | ||
|
|
9fc0ff5953 | ||
|
|
997d09bb33 | ||
|
|
b244b4ec21 | ||
|
|
b872ffee01 | ||
|
|
9a9543013c | ||
|
|
b98ce6c84f | ||
|
|
7b43b6396f | ||
|
|
67f8fcb103 | ||
|
|
352a944868 | ||
|
|
77c9116c91 | ||
|
|
e821787095 | ||
|
|
aa6e33e208 | ||
|
|
176581d732 | ||
|
|
01aa078e9c | ||
|
|
149d75c092 | ||
|
|
2619bff3cf | ||
|
|
4fb54b066f | ||
|
|
e124ef57e3 | ||
|
|
73e6b2550b | ||
|
|
241113200e | ||
|
|
6f3ab09319 | ||
|
|
3b5d5014e0 | ||
|
|
c36c3cb470 |
39
.github/workflows/build.yml
vendored
39
.github/workflows/build.yml
vendored
@@ -8,11 +8,27 @@ jobs:
|
||||
strategy:
|
||||
fail-fast: false
|
||||
matrix:
|
||||
options: [ '-mb braces', '-mc', '--toplevel -mc passes=10,pure_getters,unsafe' ]
|
||||
script: [ acorn.sh, buble.sh, butternut.sh, mathjs.sh, rollup-es.sh, rollup-ts.sh, sucrase.sh ]
|
||||
options:
|
||||
- '-mb braces'
|
||||
- '--ie -c'
|
||||
- '-mc'
|
||||
- '-p acorn --toplevel -mco spidermonkey'
|
||||
- '--toplevel -mc passes=3,pure_getters,unsafe'
|
||||
script:
|
||||
- acorn.sh
|
||||
- bootstrap.sh
|
||||
- buble.sh
|
||||
- butternut.sh
|
||||
- mathjs.sh
|
||||
- rollup-es.sh
|
||||
- rollup-ts.sh
|
||||
- sucrase.sh
|
||||
- web-tooling-benchmark.sh
|
||||
include:
|
||||
- node: '14'
|
||||
script: acorn.sh
|
||||
- node: '14'
|
||||
script: bootstrap.sh
|
||||
- node: '14'
|
||||
script: buble.sh
|
||||
- node: '14'
|
||||
@@ -25,6 +41,8 @@ jobs:
|
||||
script: rollup-ts.sh
|
||||
- node: '14'
|
||||
script: sucrase.sh
|
||||
- node: '14'
|
||||
script: web-tooling-benchmark.sh
|
||||
name: ${{ matrix.script }} ${{ matrix.options }}
|
||||
runs-on: ubuntu-latest
|
||||
env:
|
||||
@@ -36,20 +54,5 @@ jobs:
|
||||
- name: Perform uglify, build & test
|
||||
shell: bash
|
||||
run: |
|
||||
git clone --branch v1.6.0 --depth 1 https://github.com/jasongin/nvs.git ~/.nvs
|
||||
while ! timeout 60 bash -c '. ~/.nvs/nvs.sh add $NODE && nvs use $NODE'; do
|
||||
cd ~/.nvs
|
||||
while !(git clean -xdf); do echo "'git clean' failed - retrying..."; done
|
||||
cd -
|
||||
done
|
||||
. ~/.nvs/nvs.sh --version
|
||||
nvs use $NODE
|
||||
node --version
|
||||
npm config set audit false
|
||||
npm config set optional false
|
||||
npm config set save false
|
||||
npm config set strict-ssl false
|
||||
npm config set update-notifier false
|
||||
npm --version
|
||||
while !(npm install); do echo "'npm install' failed - retrying..."; done
|
||||
. ./test/release/install.sh
|
||||
./test/release/$SCRIPT $OPTIONS
|
||||
|
||||
17
.github/workflows/ci.yml
vendored
17
.github/workflows/ci.yml
vendored
@@ -29,20 +29,5 @@ jobs:
|
||||
- name: Perform tests
|
||||
shell: bash
|
||||
run: |
|
||||
git clone --branch v1.6.0 --depth 1 https://github.com/jasongin/nvs.git ~/.nvs
|
||||
while ! timeout 60 bash -c '. ~/.nvs/nvs.sh add $NODE && nvs use $NODE'; do
|
||||
cd ~/.nvs
|
||||
while !(git clean -xdf); do echo "'git clean' failed - retrying..."; done
|
||||
cd -
|
||||
done
|
||||
. ~/.nvs/nvs.sh --version
|
||||
nvs use $NODE
|
||||
node --version
|
||||
npm config set audit false
|
||||
npm config set optional false
|
||||
npm config set save false
|
||||
npm config set strict-ssl false
|
||||
npm config set update-notifier false
|
||||
npm --version
|
||||
while !(npm install); do echo "'npm install' failed - retrying..."; done
|
||||
. ./test/release/install.sh
|
||||
node test/$TYPE
|
||||
|
||||
40
.github/workflows/ufuzz.yml
vendored
40
.github/workflows/ufuzz.yml
vendored
@@ -2,11 +2,14 @@ name: Fuzzing
|
||||
on:
|
||||
pull_request:
|
||||
schedule:
|
||||
- cron: '*/5 * * * *'
|
||||
- cron: '*/15 * * * *'
|
||||
workflow_dispatch:
|
||||
workflow_run:
|
||||
branches: [ master ]
|
||||
types: [ completed ]
|
||||
workflows: [ 'Build testing', CI ]
|
||||
env:
|
||||
BASE_URL: https://api.github.com/repos/${{ github.repository }}
|
||||
CAUSE: ${{ github.event_name }}
|
||||
RUN_NUM: ${{ github.run_number }}
|
||||
TOKEN: ${{ github.token }}
|
||||
jobs:
|
||||
ufuzz:
|
||||
@@ -30,35 +33,12 @@ jobs:
|
||||
NODE: ${{ matrix.node }}
|
||||
steps:
|
||||
- uses: actions/checkout@v2
|
||||
- name: Install GNU Core Utilities
|
||||
if: ${{ startsWith(matrix.os, 'macos') }}
|
||||
env:
|
||||
HOMEBREW_NO_BOTTLE_SOURCE_FALLBACK: 1
|
||||
HOMEBREW_NO_INSTALL_CLEANUP: 1
|
||||
shell: bash
|
||||
run: |
|
||||
while !(brew install coreutils); do echo "'brew install' failed - retrying..."; done
|
||||
- name: Perform fuzzing
|
||||
shell: bash
|
||||
run: |
|
||||
git clone --branch v1.6.0 --depth 1 https://github.com/jasongin/nvs.git ~/.nvs
|
||||
while ! timeout 60 bash -c '. ~/.nvs/nvs.sh add $NODE && nvs use $NODE'; do
|
||||
cd ~/.nvs
|
||||
while !(git clean -xdf); do echo "'git clean' failed - retrying..."; done
|
||||
cd -
|
||||
done
|
||||
. ~/.nvs/nvs.sh --version
|
||||
nvs use $NODE
|
||||
node --version
|
||||
npm config set audit false
|
||||
npm config set optional false
|
||||
npm config set save false
|
||||
npm config set strict-ssl false
|
||||
npm config set update-notifier false
|
||||
npm --version
|
||||
while !(npm install); do echo "'npm install' failed - retrying..."; done
|
||||
if [[ $CAUSE == "schedule" ]]; then
|
||||
node test/ufuzz/job $BASE_URL $TOKEN $RUN_NUM
|
||||
else
|
||||
. ./test/release/install.sh
|
||||
if [[ $GITHUB_EVENT_NAME == "pull_request" ]]; then
|
||||
node test/ufuzz/job 5000
|
||||
else
|
||||
node test/ufuzz/job $BASE_URL $TOKEN $GITHUB_RUN_NUMBER
|
||||
fi
|
||||
|
||||
309
README.md
309
README.md
@@ -92,6 +92,9 @@ a double dash to prevent input files being used as option arguments:
|
||||
-o, --output <file> Output file path (default STDOUT). Specify `ast` or
|
||||
`spidermonkey` to write UglifyJS or SpiderMonkey AST
|
||||
as JSON to STDOUT respectively.
|
||||
--annotations Process and preserve comment annotations.
|
||||
(`/*@__PURE__*/` or `/*#__PURE__*/`)
|
||||
--no-annotations Ignore and discard comment annotations.
|
||||
--comments [filter] Preserve copyright comments in the output. By
|
||||
default this works like Google Closure, keeping
|
||||
JSDoc-style comments that contain "@license" or
|
||||
@@ -108,8 +111,8 @@ a double dash to prevent input files being used as option arguments:
|
||||
-d, --define <expr>[=value] Global definitions.
|
||||
-e, --enclose [arg[:value]] Embed everything in a big function, with configurable
|
||||
argument(s) & value(s).
|
||||
--ie8 Support non-standard Internet Explorer 8.
|
||||
Equivalent to setting `ie8: true` in `minify()`
|
||||
--ie Support non-standard Internet Explorer.
|
||||
Equivalent to setting `ie: true` in `minify()`
|
||||
for `compress`, `mangle` and `output` options.
|
||||
By default UglifyJS will not try to be IE-proof.
|
||||
--keep-fnames Do not mangle/drop function names. Useful for
|
||||
@@ -221,10 +224,10 @@ Example:
|
||||
To enable the mangler you need to pass `--mangle` (`-m`). The following
|
||||
(comma-separated) options are supported:
|
||||
|
||||
- `eval` (default `false`) -- mangle names visible in scopes where `eval` or
|
||||
- `eval` (default: `false`) — mangle names visible in scopes where `eval` or
|
||||
`with` are used.
|
||||
|
||||
- `reserved` (default: `[]`) -- when mangling is enabled but you want to
|
||||
- `reserved` (default: `[]`) — when mangling is enabled but you want to
|
||||
prevent certain names from being mangled, you can declare those names with
|
||||
`--mangle reserved` — pass a comma-separated list of names. For example:
|
||||
|
||||
@@ -491,46 +494,51 @@ if (result.error) throw result.error;
|
||||
|
||||
## Minify options
|
||||
|
||||
- `compress` (default `{}`) — pass `false` to skip compressing entirely.
|
||||
- `annotations` — pass `false` to ignore all comment annotations and elide them
|
||||
from output. Useful when, for instance, external tools incorrectly applied
|
||||
`/*@__PURE__*/` or `/*#__PURE__*/`. Pass `true` to both compress and retain
|
||||
comment annotations in output to allow for further processing downstream.
|
||||
|
||||
- `compress` (default: `{}`) — pass `false` to skip compressing entirely.
|
||||
Pass an object to specify custom [compress options](#compress-options).
|
||||
|
||||
- `ie8` (default `false`) -- set to `true` to support IE8.
|
||||
- `ie` (default: `false`) — enable workarounds for Internet Explorer bugs.
|
||||
|
||||
- `keep_fnames` (default: `false`) -- pass `true` to prevent discarding or mangling
|
||||
- `keep_fnames` (default: `false`) — pass `true` to prevent discarding or mangling
|
||||
of function names. Useful for code relying on `Function.prototype.name`.
|
||||
|
||||
- `mangle` (default `true`) — pass `false` to skip mangling names, or pass
|
||||
- `mangle` (default: `true`) — pass `false` to skip mangling names, or pass
|
||||
an object to specify [mangle options](#mangle-options) (see below).
|
||||
|
||||
- `mangle.properties` (default `false`) — a subcategory of the mangle option.
|
||||
- `mangle.properties` (default: `false`) — a subcategory of the mangle option.
|
||||
Pass an object to specify custom [mangle property options](#mangle-properties-options).
|
||||
|
||||
- `nameCache` (default `null`) -- pass an empty object `{}` or a previously
|
||||
- `nameCache` (default: `null`) — pass an empty object `{}` or a previously
|
||||
used `nameCache` object if you wish to cache mangled variable and
|
||||
property names across multiple invocations of `minify()`. Note: this is
|
||||
a read/write property. `minify()` will read the name cache state of this
|
||||
object and update it during minification so that it may be
|
||||
reused or externally persisted by the user.
|
||||
|
||||
- `output` (default `null`) — pass an object if you wish to specify
|
||||
- `output` (default: `null`) — pass an object if you wish to specify
|
||||
additional [output options](#output-options). The defaults are optimized
|
||||
for best compression.
|
||||
|
||||
- `parse` (default `{}`) — pass an object if you wish to specify some
|
||||
- `parse` (default: `{}`) — pass an object if you wish to specify some
|
||||
additional [parse options](#parse-options).
|
||||
|
||||
- `sourceMap` (default `false`) -- pass an object if you wish to specify
|
||||
- `sourceMap` (default: `false`) — pass an object if you wish to specify
|
||||
[source map options](#source-map-options).
|
||||
|
||||
- `toplevel` (default `false`) -- set to `true` if you wish to enable top level
|
||||
- `toplevel` (default: `false`) — set to `true` if you wish to enable top level
|
||||
variable and function name mangling and to drop unused variables and functions.
|
||||
|
||||
- `v8` (default `false`) -- enable workarounds for Chrome & Node.js bugs.
|
||||
- `v8` (default: `false`) — enable workarounds for Chrome & Node.js bugs.
|
||||
|
||||
- `warnings` (default `false`) — pass `true` to return compressor warnings
|
||||
- `warnings` (default: `false`) — pass `true` to return compressor warnings
|
||||
in `result.warnings`. Use the value `"verbose"` for more detailed warnings.
|
||||
|
||||
- `webkit` (default `false`) -- enable workarounds for Safari/WebKit bugs.
|
||||
- `webkit` (default: `false`) — enable workarounds for Safari/WebKit bugs.
|
||||
PhantomJS users should set this option to `true`.
|
||||
|
||||
## Minify options structure
|
||||
@@ -558,7 +566,6 @@ if (result.error) throw result.error;
|
||||
},
|
||||
nameCache: null, // or specify a name cache object
|
||||
toplevel: false,
|
||||
ie8: false,
|
||||
warnings: false,
|
||||
}
|
||||
```
|
||||
@@ -615,116 +622,121 @@ to be `false` and all symbol names will be omitted.
|
||||
|
||||
## Parse options
|
||||
|
||||
- `bare_returns` (default `false`) -- support top level `return` statements
|
||||
- `bare_returns` (default: `false`) — support top level `return` statements
|
||||
|
||||
- `html5_comments` (default `true`)
|
||||
- `html5_comments` (default: `true`)
|
||||
|
||||
- `shebang` (default `true`) -- support `#!command` as the first line
|
||||
- `shebang` (default: `true`) — support `#!command` as the first line
|
||||
|
||||
## Compress options
|
||||
|
||||
- `arguments` (default: `true`) -- replace `arguments[index]` with function
|
||||
- `annotations` (default: `true`) — Pass `false` to disable potentially dropping
|
||||
functions marked as "pure". A function call is marked as "pure" if a comment
|
||||
annotation `/*@__PURE__*/` or `/*#__PURE__*/` immediately precedes the call. For
|
||||
example: `/*@__PURE__*/foo();`
|
||||
|
||||
- `arguments` (default: `true`) — replace `arguments[index]` with function
|
||||
parameter name whenever possible.
|
||||
|
||||
- `arrows` (default: `true`) -- apply optimizations to arrow functions
|
||||
- `arrows` (default: `true`) — apply optimizations to arrow functions
|
||||
|
||||
- `assignments` (default: `true`) -- apply optimizations to assignment expressions
|
||||
- `assignments` (default: `true`) — apply optimizations to assignment expressions
|
||||
|
||||
- `awaits` (default: `true`) -- apply optimizations to `await` expressions
|
||||
- `awaits` (default: `true`) — apply optimizations to `await` expressions
|
||||
|
||||
- `booleans` (default: `true`) -- various optimizations for boolean context,
|
||||
- `booleans` (default: `true`) — various optimizations for boolean context,
|
||||
for example `!!a ? b : c → a ? b : c`
|
||||
|
||||
- `collapse_vars` (default: `true`) -- Collapse single-use non-constant variables,
|
||||
- `collapse_vars` (default: `true`) — Collapse single-use non-constant variables,
|
||||
side effects permitting.
|
||||
|
||||
- `comparisons` (default: `true`) -- apply certain optimizations to binary nodes,
|
||||
- `comparisons` (default: `true`) — apply certain optimizations to binary nodes,
|
||||
e.g. `!(a <= b) → a > b`, attempts to negate binary nodes, e.g.
|
||||
`a = !b && !c && !d && !e → a=!(b||c||d||e)` etc.
|
||||
|
||||
- `conditionals` (default: `true`) -- apply optimizations for `if`-s and conditional
|
||||
- `conditionals` (default: `true`) — apply optimizations for `if`-s and conditional
|
||||
expressions
|
||||
|
||||
- `dead_code` (default: `true`) -- remove unreachable code
|
||||
- `dead_code` (default: `true`) — remove unreachable code
|
||||
|
||||
- `default_values` (default: `true`) -- drop overshadowed default values
|
||||
- `default_values` (default: `true`) — drop overshadowed default values
|
||||
|
||||
- `directives` (default: `true`) -- remove redundant or non-standard directives
|
||||
- `directives` (default: `true`) — remove redundant or non-standard directives
|
||||
|
||||
- `drop_console` (default: `false`) -- Pass `true` to discard calls to
|
||||
- `drop_console` (default: `false`) — Pass `true` to discard calls to
|
||||
`console.*` functions. If you wish to drop a specific function call
|
||||
such as `console.info` and/or retain side effects from function arguments
|
||||
after dropping the function call then use `pure_funcs` instead.
|
||||
|
||||
- `drop_debugger` (default: `true`) -- remove `debugger;` statements
|
||||
- `drop_debugger` (default: `true`) — remove `debugger;` statements
|
||||
|
||||
- `evaluate` (default: `true`) -- Evaluate expression for shorter constant
|
||||
- `evaluate` (default: `true`) — Evaluate expression for shorter constant
|
||||
representation. Pass `"eager"` to always replace function calls whenever
|
||||
possible, or a positive integer to specify an upper bound for each individual
|
||||
evaluation in number of characters.
|
||||
|
||||
- `expression` (default: `false`) -- Pass `true` to preserve completion values
|
||||
- `expression` (default: `false`) — Pass `true` to preserve completion values
|
||||
from terminal statements without `return`, e.g. in bookmarklets.
|
||||
|
||||
- `functions` (default: `true`) -- convert declarations from `var` to `function`
|
||||
- `functions` (default: `true`) — convert declarations from `var` to `function`
|
||||
whenever possible.
|
||||
|
||||
- `global_defs` (default: `{}`) -- see [conditional compilation](#conditional-compilation)
|
||||
- `global_defs` (default: `{}`) — see [conditional compilation](#conditional-compilation)
|
||||
|
||||
- `hoist_exports` (default: `true`) -- hoist `export` statements to facilitate
|
||||
- `hoist_exports` (default: `true`) — hoist `export` statements to facilitate
|
||||
various `compress` and `mangle` optimizations.
|
||||
|
||||
- `hoist_funs` (default: `false`) -- hoist function declarations
|
||||
- `hoist_funs` (default: `false`) — hoist function declarations
|
||||
|
||||
- `hoist_props` (default: `true`) -- hoist properties from constant object and
|
||||
- `hoist_props` (default: `true`) — hoist properties from constant object and
|
||||
array literals into regular variables subject to a set of constraints. For example:
|
||||
`var o={p:1, q:2}; f(o.p, o.q);` is converted to `f(1, 2);`. Note: `hoist_props`
|
||||
works best with `toplevel` and `mangle` enabled, alongside with `compress` option
|
||||
`passes` set to `2` or higher.
|
||||
|
||||
- `hoist_vars` (default: `false`) -- hoist `var` declarations (this is `false`
|
||||
- `hoist_vars` (default: `false`) — hoist `var` declarations (this is `false`
|
||||
by default because it seems to increase the size of the output in general)
|
||||
|
||||
- `if_return` (default: `true`) -- optimizations for if/return and if/continue
|
||||
- `if_return` (default: `true`) — optimizations for if/return and if/continue
|
||||
|
||||
- `imports` (default: `true`) -- drop unreferenced import symbols when used with `unused`
|
||||
- `imports` (default: `true`) — drop unreferenced import symbols when used with `unused`
|
||||
|
||||
- `inline` (default: `true`) -- inline calls to function with simple/`return` statement:
|
||||
- `false` -- same as `0`
|
||||
- `0` -- disabled inlining
|
||||
- `1` -- inline simple functions
|
||||
- `2` -- inline functions with arguments
|
||||
- `3` -- inline functions with arguments and variables
|
||||
- `true` -- same as `3`
|
||||
- `inline` (default: `true`) — inline calls to function with simple/`return` statement:
|
||||
- `false` — same as `0`
|
||||
- `0` — disabled inlining
|
||||
- `1` — inline simple functions
|
||||
- `2` — inline functions with arguments
|
||||
- `3` — inline functions with arguments and variables
|
||||
- `true` — same as `3`
|
||||
|
||||
- `join_vars` (default: `true`) -- join consecutive `var` statements
|
||||
- `join_vars` (default: `true`) — join consecutive `var` statements
|
||||
|
||||
- `keep_fargs` (default: `false`) -- discard unused function arguments except
|
||||
- `keep_fargs` (default: `false`) — discard unused function arguments except
|
||||
when unsafe to do so, e.g. code which relies on `Function.prototype.length`.
|
||||
Pass `true` to always retain function arguments.
|
||||
|
||||
- `keep_infinity` (default: `false`) -- Pass `true` to prevent `Infinity` from
|
||||
- `keep_infinity` (default: `false`) — Pass `true` to prevent `Infinity` from
|
||||
being compressed into `1/0`, which may cause performance issues on Chrome.
|
||||
|
||||
- `loops` (default: `true`) -- optimizations for `do`, `while` and `for` loops
|
||||
- `loops` (default: `true`) — optimizations for `do`, `while` and `for` loops
|
||||
when we can statically determine the condition.
|
||||
|
||||
- `merge_vars` (default: `true`) -- combine and reuse variables.
|
||||
- `merge_vars` (default: `true`) — combine and reuse variables.
|
||||
|
||||
- `negate_iife` (default: `true`) -- negate "Immediately-Called Function Expressions"
|
||||
- `negate_iife` (default: `true`) — negate "Immediately-Called Function Expressions"
|
||||
where the return value is discarded, to avoid the parens that the
|
||||
code generator would insert.
|
||||
|
||||
- `objects` (default: `true`) -- compact duplicate keys in object literals.
|
||||
- `objects` (default: `true`) — compact duplicate keys in object literals.
|
||||
|
||||
- `passes` (default: `1`) -- The maximum number of times to run compress.
|
||||
- `passes` (default: `1`) — The maximum number of times to run compress.
|
||||
In some cases more than one pass leads to further compressed code. Keep in
|
||||
mind more passes will take more time.
|
||||
|
||||
- `properties` (default: `true`) -- rewrite property access using the dot notation, for
|
||||
- `properties` (default: `true`) — rewrite property access using the dot notation, for
|
||||
example `foo["bar"] → foo.bar`
|
||||
|
||||
- `pure_funcs` (default: `null`) -- You can pass an array of names and
|
||||
- `pure_funcs` (default: `null`) — You can pass an array of names and
|
||||
UglifyJS will assume that those functions do not produce side
|
||||
effects. DANGER: will not check if the name is redefined in scope.
|
||||
An example case here, for instance `var q = Math.floor(a/b)`. If
|
||||
@@ -736,24 +748,24 @@ to be `false` and all symbol names will be omitted.
|
||||
overhead (compression will be slower). Make sure symbols under `pure_funcs`
|
||||
are also under `mangle.reserved` to avoid mangling.
|
||||
|
||||
- `pure_getters` (default: `"strict"`) -- If you pass `true` for
|
||||
- `pure_getters` (default: `"strict"`) — If you pass `true` for
|
||||
this, UglifyJS will assume that object property access
|
||||
(e.g. `foo.bar` or `foo["bar"]`) doesn't have any side effects.
|
||||
Specify `"strict"` to treat `foo.bar` as side-effect-free only when
|
||||
`foo` is certain to not throw, i.e. not `null` or `undefined`.
|
||||
|
||||
- `reduce_funcs` (default: `true`) -- Allows single-use functions to be
|
||||
- `reduce_funcs` (default: `true`) — Allows single-use functions to be
|
||||
inlined as function expressions when permissible allowing further
|
||||
optimization. Enabled by default. Option depends on `reduce_vars`
|
||||
being enabled. Some code runs faster in the Chrome V8 engine if this
|
||||
option is disabled. Does not negatively impact other major browsers.
|
||||
|
||||
- `reduce_vars` (default: `true`) -- Improve optimization on variables assigned with and
|
||||
- `reduce_vars` (default: `true`) — Improve optimization on variables assigned with and
|
||||
used as constant values.
|
||||
|
||||
- `rests` (default: `true`) -- apply optimizations to rest parameters
|
||||
- `rests` (default: `true`) — apply optimizations to rest parameters
|
||||
|
||||
- `sequences` (default: `true`) -- join consecutive simple statements using the
|
||||
- `sequences` (default: `true`) — join consecutive simple statements using the
|
||||
comma operator. May be set to a positive integer to specify the maximum number
|
||||
of consecutive comma sequences that will be generated. If this option is set to
|
||||
`true` then the default `sequences` limit is `200`. Set option to `false` or `0`
|
||||
@@ -762,70 +774,67 @@ to be `false` and all symbol names will be omitted.
|
||||
occasions the default sequences limit leads to very slow compress times in which
|
||||
case a value of `20` or less is recommended.
|
||||
|
||||
- `side_effects` (default: `true`) -- Pass `false` to disable potentially dropping
|
||||
functions marked as "pure". A function call is marked as "pure" if a comment
|
||||
annotation `/*@__PURE__*/` or `/*#__PURE__*/` immediately precedes the call. For
|
||||
example: `/*@__PURE__*/foo();`
|
||||
- `side_effects` (default: `true`) — drop extraneous code which does not affect
|
||||
outcome of runtime execution.
|
||||
|
||||
- `spreads` (default: `true`) -- flatten spread expressions.
|
||||
- `spreads` (default: `true`) — flatten spread expressions.
|
||||
|
||||
- `strings` (default: `true`) -- compact string concatenations.
|
||||
- `strings` (default: `true`) — compact string concatenations.
|
||||
|
||||
- `switches` (default: `true`) -- de-duplicate and remove unreachable `switch` branches
|
||||
- `switches` (default: `true`) — de-duplicate and remove unreachable `switch` branches
|
||||
|
||||
- `templates` (default: `true`) -- compact template literals by embedding expressions
|
||||
- `templates` (default: `true`) — compact template literals by embedding expressions
|
||||
and/or converting to string literals, e.g. `` `foo ${42}` → "foo 42"``
|
||||
|
||||
- `top_retain` (default: `null`) -- prevent specific toplevel functions and
|
||||
- `top_retain` (default: `null`) — prevent specific toplevel functions and
|
||||
variables from `unused` removal (can be array, comma-separated, RegExp or
|
||||
function. Implies `toplevel`)
|
||||
|
||||
- `toplevel` (default: `false`) -- drop unreferenced functions (`"funcs"`) and/or
|
||||
- `toplevel` (default: `false`) — drop unreferenced functions (`"funcs"`) and/or
|
||||
variables (`"vars"`) in the top level scope (`false` by default, `true` to drop
|
||||
both unreferenced functions and variables)
|
||||
|
||||
- `typeofs` (default: `true`) -- Transforms `typeof foo == "undefined"` into
|
||||
`foo === void 0`. Note: recommend to set this value to `false` for IE10 and
|
||||
earlier versions due to known issues.
|
||||
- `typeofs` (default: `true`) — compress `typeof` expressions, e.g.
|
||||
`typeof foo == "undefined" → void 0 === foo`
|
||||
|
||||
- `unsafe` (default: `false`) -- apply "unsafe" transformations (discussion below)
|
||||
- `unsafe` (default: `false`) — apply "unsafe" transformations (discussion below)
|
||||
|
||||
- `unsafe_comps` (default: `false`) -- compress expressions like `a <= b` assuming
|
||||
- `unsafe_comps` (default: `false`) — compress expressions like `a <= b` assuming
|
||||
none of the operands can be (coerced to) `NaN`.
|
||||
|
||||
- `unsafe_Function` (default: `false`) -- compress and mangle `Function(args, code)`
|
||||
- `unsafe_Function` (default: `false`) — compress and mangle `Function(args, code)`
|
||||
when both `args` and `code` are string literals.
|
||||
|
||||
- `unsafe_math` (default: `false`) -- optimize numerical expressions like
|
||||
- `unsafe_math` (default: `false`) — optimize numerical expressions like
|
||||
`2 * x * 3` into `6 * x`, which may give imprecise floating point results.
|
||||
|
||||
- `unsafe_proto` (default: `false`) -- optimize expressions like
|
||||
- `unsafe_proto` (default: `false`) — optimize expressions like
|
||||
`Array.prototype.slice.call(a)` into `[].slice.call(a)`
|
||||
|
||||
- `unsafe_regexp` (default: `false`) -- enable substitutions of variables with
|
||||
- `unsafe_regexp` (default: `false`) — enable substitutions of variables with
|
||||
`RegExp` values the same way as if they are constants.
|
||||
|
||||
- `unsafe_undefined` (default: `false`) -- substitute `void 0` if there is a
|
||||
- `unsafe_undefined` (default: `false`) — substitute `void 0` if there is a
|
||||
variable named `undefined` in scope (variable name will be mangled, typically
|
||||
reduced to a single character)
|
||||
|
||||
- `unused` (default: `true`) -- drop unreferenced functions and variables (simple
|
||||
- `unused` (default: `true`) — drop unreferenced functions and variables (simple
|
||||
direct variable assignments do not count as references unless set to `"keep_assign"`)
|
||||
|
||||
- `varify` (default: `true`) -- convert block-scoped declaractions into `var`
|
||||
- `varify` (default: `true`) — convert block-scoped declaractions into `var`
|
||||
whenever safe to do so
|
||||
|
||||
- `yields` (default: `true`) -- apply optimizations to `yield` expressions
|
||||
- `yields` (default: `true`) — apply optimizations to `yield` expressions
|
||||
|
||||
## Mangle options
|
||||
|
||||
- `eval` (default `false`) -- Pass `true` to mangle names visible in scopes
|
||||
- `eval` (default: `false`) — Pass `true` to mangle names visible in scopes
|
||||
where `eval` or `with` are used.
|
||||
|
||||
- `reserved` (default `[]`) -- Pass an array of identifiers that should be
|
||||
- `reserved` (default: `[]`) — Pass an array of identifiers that should be
|
||||
excluded from mangling. Example: `["foo", "bar"]`.
|
||||
|
||||
- `toplevel` (default `false`) -- Pass `true` to mangle names declared in the
|
||||
- `toplevel` (default: `false`) — Pass `true` to mangle names declared in the
|
||||
top level scope.
|
||||
|
||||
Examples:
|
||||
@@ -852,18 +861,18 @@ UglifyJS.minify(code, { mangle: { toplevel: true } }).code;
|
||||
|
||||
### Mangle properties options
|
||||
|
||||
- `builtins` (default: `false`) -- Use `true` to allow the mangling of builtin
|
||||
- `builtins` (default: `false`) — Use `true` to allow the mangling of builtin
|
||||
DOM properties. Not recommended to override this setting.
|
||||
|
||||
- `debug` (default: `false`) -— Mangle names with the original name still present.
|
||||
- `debug` (default: `false`) — Mangle names with the original name still present.
|
||||
Pass an empty string `""` to enable, or a non-empty string to set the debug suffix.
|
||||
|
||||
- `keep_quoted` (default: `false`) -— Only mangle unquoted property names.
|
||||
- `keep_quoted` (default: `false`) — Only mangle unquoted property names.
|
||||
|
||||
- `regex` (default: `null`) -— Pass a RegExp literal to only mangle property
|
||||
- `regex` (default: `null`) — Pass a RegExp literal to only mangle property
|
||||
names matching the regular expression.
|
||||
|
||||
- `reserved` (default: `[]`) -- Do not mangle property names listed in the
|
||||
- `reserved` (default: `[]`) — Do not mangle property names listed in the
|
||||
`reserved` array.
|
||||
|
||||
## Output options
|
||||
@@ -872,19 +881,23 @@ The code generator tries to output shortest code possible by default. In
|
||||
case you want beautified output, pass `--beautify` (`-b`). Optionally you
|
||||
can pass additional arguments that control the code output:
|
||||
|
||||
- `ascii_only` (default `false`) -- escape Unicode characters in strings and
|
||||
- `annotations` (default: `false`) — pass `true` to retain comment annotations
|
||||
`/*@__PURE__*/` or `/*#__PURE__*/`, otherwise they will be discarded even if
|
||||
`comments` is set.
|
||||
|
||||
- `ascii_only` (default: `false`) — escape Unicode characters in strings and
|
||||
regexps (affects directives with non-ascii characters becoming invalid)
|
||||
|
||||
- `beautify` (default `true`) -- whether to actually beautify the output.
|
||||
- `beautify` (default: `true`) — whether to actually beautify the output.
|
||||
Passing `-b` will set this to true, but you might need to pass `-b` even
|
||||
when you want to generate minified code, in order to specify additional
|
||||
arguments, so you can use `-b beautify=false` to override it.
|
||||
|
||||
- `braces` (default `false`) -- always insert braces in `if`, `for`,
|
||||
- `braces` (default: `false`) — always insert braces in `if`, `for`,
|
||||
`do`, `while` or `with` statements, even if their body is a single
|
||||
statement.
|
||||
|
||||
- `comments` (default `false`) -- pass `true` or `"all"` to preserve all
|
||||
- `comments` (default: `false`) — pass `true` or `"all"` to preserve all
|
||||
comments, `"some"` to preserve multi-line comments that contain `@cc_on`,
|
||||
`@license`, or `@preserve` (case-insensitive), a regular expression string
|
||||
(e.g. `/^!/`), or a function which returns `boolean`, e.g.
|
||||
@@ -894,53 +907,53 @@ can pass additional arguments that control the code output:
|
||||
}
|
||||
```
|
||||
|
||||
- `galio` (default `false`) -- enable workarounds for ANT Galio bugs
|
||||
- `galio` (default: `false`) — enable workarounds for ANT Galio bugs
|
||||
|
||||
- `indent_level` (default `4`)
|
||||
- `indent_level` (default: `4`)
|
||||
|
||||
- `indent_start` (default `0`) -- prefix all lines by that many spaces
|
||||
- `indent_start` (default: `0`) — prefix all lines by that many spaces
|
||||
|
||||
- `inline_script` (default `true`) -- escape HTML comments and the slash in
|
||||
- `inline_script` (default: `true`) — escape HTML comments and the slash in
|
||||
occurrences of `</script>` in strings
|
||||
|
||||
- `keep_quoted_props` (default `false`) -- when turned on, prevents stripping
|
||||
- `keep_quoted_props` (default: `false`) — when turned on, prevents stripping
|
||||
quotes from property names in object literals.
|
||||
|
||||
- `max_line_len` (default `false`) -- maximum line length (for uglified code)
|
||||
- `max_line_len` (default: `false`) — maximum line length (for uglified code)
|
||||
|
||||
- `preamble` (default `null`) -- when passed it must be a string and
|
||||
- `preamble` (default: `null`) — when passed it must be a string and
|
||||
it will be prepended to the output literally. The source map will
|
||||
adjust for this text. Can be used to insert a comment containing
|
||||
licensing information, for example.
|
||||
|
||||
- `preserve_line` (default `false`) -- pass `true` to retain line numbering on
|
||||
- `preserve_line` (default: `false`) — pass `true` to retain line numbering on
|
||||
a best effort basis.
|
||||
|
||||
- `quote_keys` (default `false`) -- pass `true` to quote all keys in literal
|
||||
- `quote_keys` (default: `false`) — pass `true` to quote all keys in literal
|
||||
objects
|
||||
|
||||
- `quote_style` (default `0`) -- preferred quote style for strings (affects
|
||||
- `quote_style` (default: `0`) — preferred quote style for strings (affects
|
||||
quoted property names and directives as well):
|
||||
- `0` -- prefers double quotes, switches to single quotes when there are
|
||||
- `0` — prefers double quotes, switches to single quotes when there are
|
||||
more double quotes in the string itself. `0` is best for gzip size.
|
||||
- `1` -- always use single quotes
|
||||
- `2` -- always use double quotes
|
||||
- `3` -- always use the original quotes
|
||||
- `1` — always use single quotes
|
||||
- `2` — always use double quotes
|
||||
- `3` — always use the original quotes
|
||||
|
||||
- `semicolons` (default `true`) -- separate statements with semicolons. If
|
||||
- `semicolons` (default: `true`) — separate statements with semicolons. If
|
||||
you pass `false` then whenever possible we will use a newline instead of a
|
||||
semicolon, leading to more readable output of uglified code (size before
|
||||
gzip could be smaller; size after gzip insignificantly larger).
|
||||
|
||||
- `shebang` (default `true`) -- preserve shebang `#!` in preamble (bash scripts)
|
||||
- `shebang` (default: `true`) — preserve shebang `#!` in preamble (bash scripts)
|
||||
|
||||
- `width` (default `80`) -- only takes effect when beautification is on, this
|
||||
- `width` (default: `80`) — only takes effect when beautification is on, this
|
||||
specifies an (orientative) line width that the beautifier will try to
|
||||
obey. It refers to the width of the line text (excluding indentation).
|
||||
It doesn't work very well currently, but it does make the code generated
|
||||
by UglifyJS more readable.
|
||||
|
||||
- `wrap_iife` (default `false`) -- pass `true` to wrap immediately invoked
|
||||
- `wrap_iife` (default: `false`) — pass `true` to wrap immediately invoked
|
||||
function expressions. See
|
||||
[#640](https://github.com/mishoo/UglifyJS/issues/640) for more details.
|
||||
|
||||
@@ -1281,3 +1294,61 @@ To allow for better optimizations, the compiler makes various assumptions:
|
||||
// SyntaxError: Identifier 'e' has already been declared
|
||||
```
|
||||
UglifyJS may modify the input which in turn may suppress those errors.
|
||||
- Some versions of Chrome and Node.js will give incorrect results with the
|
||||
following:
|
||||
```javascript
|
||||
console.log({
|
||||
...{
|
||||
set 42(v) {},
|
||||
42: "PASS",
|
||||
},
|
||||
});
|
||||
// Expected: { '42': 'PASS' }
|
||||
// Actual: { '42': undefined }
|
||||
```
|
||||
UglifyJS may modify the input which in turn may suppress those errors.
|
||||
- Later versions of JavaScript will throw `SyntaxError` with the following:
|
||||
```javascript
|
||||
var await;
|
||||
async function f() {
|
||||
class A {
|
||||
static p = await;
|
||||
}
|
||||
}
|
||||
// SyntaxError: Unexpected reserved word
|
||||
```
|
||||
UglifyJS may modify the input which in turn may suppress those errors.
|
||||
- Later versions of JavaScript will throw `SyntaxError` with the following:
|
||||
```javascript
|
||||
var async;
|
||||
for (async of []);
|
||||
// SyntaxError: The left-hand side of a for-of loop may not be 'async'.
|
||||
```
|
||||
UglifyJS may modify the input which in turn may suppress those errors.
|
||||
- Later versions of Chrome and Node.js will give incorrect results with the
|
||||
following:
|
||||
```javascript
|
||||
console.log({
|
||||
...console,
|
||||
get 42() {
|
||||
return "FAIL";
|
||||
},
|
||||
[42]: "PASS",
|
||||
}[42]);
|
||||
// Expected: "PASS"
|
||||
// Actual: "FAIL"
|
||||
```
|
||||
UglifyJS may modify the input which in turn may suppress those errors.
|
||||
- Earlier versions of JavaScript will throw `TypeError` with the following:
|
||||
```javascript
|
||||
(function() {
|
||||
{
|
||||
const a = "foo";
|
||||
}
|
||||
{
|
||||
const a = "bar";
|
||||
}
|
||||
})();
|
||||
// TypeError: const 'a' has already been declared
|
||||
```
|
||||
UglifyJS may modify the input which in turn may suppress those errors.
|
||||
|
||||
79
bin/uglifyjs
79
bin/uglifyjs
@@ -95,11 +95,13 @@ function process_option(name, no_value) {
|
||||
" -b, --beautify [options] Beautify output/specify output options.",
|
||||
" -O, --output-opts <options> Output options (beautify disabled).",
|
||||
" -o, --output <file> Output file (default STDOUT).",
|
||||
" --annotations Process and preserve comment annotations.",
|
||||
" --no-annotations Ignore and discard comment annotations.",
|
||||
" --comments [filter] Preserve copyright comments in the output.",
|
||||
" --config-file <file> Read minify() options from JSON file.",
|
||||
" -d, --define <expr>[=value] Global definitions.",
|
||||
" -e, --enclose [arg[,...][:value[,...]]] Embed everything in a big function, with configurable argument(s) & value(s).",
|
||||
" --ie8 Support non-standard Internet Explorer 8.",
|
||||
" --ie Support non-standard Internet Explorer.",
|
||||
" --keep-fnames Do not mangle/drop function names. Useful for code relying on Function.prototype.name.",
|
||||
" --name-cache <file> File to hold mangled name mappings.",
|
||||
" --rename Force symbol expansion.",
|
||||
@@ -108,6 +110,7 @@ function process_option(name, no_value) {
|
||||
" --source-map [options] Enable source map/specify source map options.",
|
||||
" --timings Display operations run time on STDERR.",
|
||||
" --toplevel Compress and/or mangle variables in toplevel scope.",
|
||||
" --v8 Support non-standard Chrome & Node.js.",
|
||||
" --validate Perform validation during AST manipulations.",
|
||||
" --verbose Print diagnostic messages.",
|
||||
" --warn Print warning messages.",
|
||||
@@ -142,13 +145,19 @@ function process_option(name, no_value) {
|
||||
case "enclose":
|
||||
options[name] = read_value();
|
||||
break;
|
||||
case "annotations":
|
||||
case "ie":
|
||||
case "ie8":
|
||||
case "timings":
|
||||
case "toplevel":
|
||||
case "v8":
|
||||
case "validate":
|
||||
case "webkit":
|
||||
options[name] = true;
|
||||
break;
|
||||
case "no-annotations":
|
||||
options.annotations = false;
|
||||
break;
|
||||
case "keep-fnames":
|
||||
options.keep_fnames = true;
|
||||
break;
|
||||
@@ -229,10 +238,11 @@ if (options.mangle && options.mangle.properties) {
|
||||
});
|
||||
}
|
||||
}
|
||||
if (output == "ast") options.output = {
|
||||
ast: true,
|
||||
code: false,
|
||||
};
|
||||
if (/^ast|spidermonkey$/.test(output)) {
|
||||
if (typeof options.output != "object") options.output = {};
|
||||
options.output.ast = true;
|
||||
options.output.code = false;
|
||||
}
|
||||
if (options.parse && (options.parse.acorn || options.parse.spidermonkey)
|
||||
&& options.sourceMap && options.sourceMap.content == "inline") {
|
||||
fatal("inline source map only works with built-in parser");
|
||||
@@ -259,7 +269,9 @@ if (specified["self"]) {
|
||||
paths = UglifyJS.FILES;
|
||||
}
|
||||
if (specified["in-situ"]) {
|
||||
if (output || specified["reduce-test"] || specified["self"]) fatal("incompatible options specified");
|
||||
if (output && output != "spidermonkey" || specified["reduce-test"] || specified["self"]) {
|
||||
fatal("incompatible options specified");
|
||||
}
|
||||
paths.forEach(function(name) {
|
||||
print(name);
|
||||
if (/^ast|spidermonkey$/.test(name)) fatal("invalid file name specified");
|
||||
@@ -305,13 +317,43 @@ function run() {
|
||||
try {
|
||||
if (options.parse) {
|
||||
if (options.parse.acorn) {
|
||||
var annotations = Object.create(null);
|
||||
files = convert_ast(function(toplevel, name) {
|
||||
return require("acorn").parse(files[name], {
|
||||
var content = files[name];
|
||||
var list = annotations[name] = [];
|
||||
var prev = -1;
|
||||
return require("acorn").parse(content, {
|
||||
allowHashBang: true,
|
||||
ecmaVersion: "latest",
|
||||
locations: true,
|
||||
onComment: function(block, text, start, end) {
|
||||
var match = /[@#]__PURE__/.exec(text);
|
||||
if (!match) {
|
||||
if (start != prev) return;
|
||||
match = [ list[prev] ];
|
||||
}
|
||||
while (/\s/.test(content[end])) end++;
|
||||
list[end] = match[0];
|
||||
prev = end;
|
||||
},
|
||||
preserveParens: true,
|
||||
program: toplevel,
|
||||
sourceFile: name
|
||||
sourceFile: name,
|
||||
sourceType: "module",
|
||||
});
|
||||
});
|
||||
files.walk(new UglifyJS.TreeWalker(function(node) {
|
||||
if (!(node instanceof UglifyJS.AST_Call)) return;
|
||||
var list = annotations[node.start.file];
|
||||
var pure = list[node.start.pos];
|
||||
if (!pure) {
|
||||
var tokens = node.start.parens;
|
||||
if (tokens) for (var i = 0; !pure && i < tokens.length; i++) {
|
||||
pure = list[tokens[i].pos];
|
||||
}
|
||||
}
|
||||
if (pure) node.pure = pure;
|
||||
}));
|
||||
} else if (options.parse.spidermonkey) {
|
||||
files = convert_ast(function(toplevel, name) {
|
||||
var obj = JSON.parse(files[name]);
|
||||
@@ -403,16 +445,19 @@ function run() {
|
||||
return value;
|
||||
}, 2));
|
||||
} else if (output == "spidermonkey") {
|
||||
print(JSON.stringify(UglifyJS.minify(result.code, {
|
||||
compress: false,
|
||||
mangle: false,
|
||||
output: {
|
||||
ast: true,
|
||||
code: false
|
||||
},
|
||||
}).ast.to_mozilla_ast(), null, 2));
|
||||
print(JSON.stringify(result.ast.to_mozilla_ast(), null, 2));
|
||||
} else if (output) {
|
||||
fs.writeFileSync(output, result.code);
|
||||
var code;
|
||||
if (result.ast) {
|
||||
var opts = {};
|
||||
for (var name in options.output) {
|
||||
if (!/^ast|code$/.test(name)) opts[name] = options.output[name];
|
||||
}
|
||||
code = UglifyJS.AST_Node.from_mozilla_ast(result.ast.to_mozilla_ast()).print_to_string(opts);
|
||||
} else {
|
||||
code = result.code;
|
||||
}
|
||||
fs.writeFileSync(output, code);
|
||||
if (result.map) fs.writeFileSync(output + ".map", result.map);
|
||||
} else {
|
||||
print(result.code);
|
||||
|
||||
92
lib/ast.js
92
lib/ast.js
@@ -55,35 +55,32 @@ function DEFNODE(type, props, methods, base) {
|
||||
props.forEach(function(prop) {
|
||||
code.push("this.", prop, "=props.", prop, ";");
|
||||
});
|
||||
var proto = base && new base;
|
||||
if (proto && proto.initialize || methods && methods.initialize) code.push("this.initialize();");
|
||||
code.push("}}");
|
||||
code.push("}");
|
||||
var proto = Object.create(base && base.prototype);
|
||||
if (methods.initialize || proto.initialize) code.push("this.initialize();");
|
||||
code.push("};");
|
||||
var ctor = new Function(code.join(""))();
|
||||
if (proto) {
|
||||
ctor.prototype = proto;
|
||||
ctor.BASE = base;
|
||||
}
|
||||
if (base) base.SUBCLASSES.push(ctor);
|
||||
ctor.prototype = proto;
|
||||
ctor.prototype.CTOR = ctor;
|
||||
ctor.PROPS = props || null;
|
||||
ctor.SELF_PROPS = self_props;
|
||||
ctor.SUBCLASSES = [];
|
||||
if (type) {
|
||||
ctor.prototype.TYPE = ctor.TYPE = type;
|
||||
}
|
||||
if (methods) for (var name in methods) if (HOP(methods, name)) {
|
||||
if (/^\$/.test(name)) {
|
||||
ctor[name.substr(1)] = methods[name];
|
||||
} else {
|
||||
ctor.prototype[name] = methods[name];
|
||||
}
|
||||
ctor.prototype.TYPE = ctor.TYPE = type;
|
||||
if (base) {
|
||||
ctor.BASE = base;
|
||||
base.SUBCLASSES.push(ctor);
|
||||
}
|
||||
ctor.DEFMETHOD = function(name, method) {
|
||||
this.prototype[name] = method;
|
||||
};
|
||||
if (typeof exports !== "undefined") {
|
||||
exports["AST_" + type] = ctor;
|
||||
ctor.PROPS = props;
|
||||
ctor.SELF_PROPS = self_props;
|
||||
ctor.SUBCLASSES = [];
|
||||
for (var name in methods) if (HOP(methods, name)) {
|
||||
if (/^\$/.test(name)) {
|
||||
ctor[name.substr(1)] = methods[name];
|
||||
} else {
|
||||
ctor.DEFMETHOD(name, methods[name]);
|
||||
}
|
||||
}
|
||||
if (typeof exports !== "undefined") exports["AST_" + type] = ctor;
|
||||
return ctor;
|
||||
}
|
||||
|
||||
@@ -260,9 +257,9 @@ var AST_BlockScope = DEFNODE("BlockScope", "enclosed functions make_def parent_s
|
||||
$documentation: "Base class for all statements introducing a lexical scope",
|
||||
$propdoc: {
|
||||
enclosed: "[SymbolDef*/S] a list of all symbol definitions that are accessed from this scope or any subscopes",
|
||||
functions: "[Object/S] like `variables`, but only lists function declarations",
|
||||
functions: "[Dictionary/S] like `variables`, but only lists function declarations",
|
||||
parent_scope: "[AST_Scope?/S] link to the parent scope",
|
||||
variables: "[Object/S] a map of name ---> SymbolDef for all variables/functions defined in this scope",
|
||||
variables: "[Dictionary/S] a map of name ---> SymbolDef for all variables/functions defined in this scope",
|
||||
},
|
||||
clone: function(deep) {
|
||||
var node = this._clone(deep);
|
||||
@@ -505,7 +502,7 @@ var AST_Scope = DEFNODE("Scope", "uses_eval uses_with", {
|
||||
var AST_Toplevel = DEFNODE("Toplevel", "globals", {
|
||||
$documentation: "The toplevel scope",
|
||||
$propdoc: {
|
||||
globals: "[Object/S] a map of name ---> SymbolDef for all undeclared names",
|
||||
globals: "[Dictionary/S] a map of name ---> SymbolDef for all undeclared names",
|
||||
},
|
||||
wrap: function(name) {
|
||||
var body = this.body;
|
||||
@@ -1292,11 +1289,14 @@ function must_be_expressions(node, prop, allow_spread, allow_hole) {
|
||||
});
|
||||
}
|
||||
|
||||
var AST_Call = DEFNODE("Call", "expression args pure", {
|
||||
var AST_Call = DEFNODE("Call", "args expression optional pure terminal", {
|
||||
$documentation: "A function call expression",
|
||||
$propdoc: {
|
||||
args: "[AST_Node*] array of arguments",
|
||||
expression: "[AST_Node] expression to invoke as function",
|
||||
args: "[AST_Node*] array of arguments"
|
||||
optional: "[boolean] whether the expression is optional chaining",
|
||||
pure: "[string/S] marker for side-effect-free call expression",
|
||||
terminal: "[boolean] whether the chain has ended",
|
||||
},
|
||||
walk: function(visitor) {
|
||||
var node = this;
|
||||
@@ -1314,7 +1314,11 @@ var AST_Call = DEFNODE("Call", "expression args pure", {
|
||||
});
|
||||
|
||||
var AST_New = DEFNODE("New", null, {
|
||||
$documentation: "An object instantiation. Derives from a function call since it has exactly the same properties"
|
||||
$documentation: "An object instantiation. Derives from a function call since it has exactly the same properties",
|
||||
_validate: function() {
|
||||
if (this.optional) throw new Error("optional must be false");
|
||||
if (this.terminal) throw new Error("terminal must be false");
|
||||
},
|
||||
}, AST_Call);
|
||||
|
||||
var AST_Sequence = DEFNODE("Sequence", "expressions", {
|
||||
@@ -1336,22 +1340,23 @@ var AST_Sequence = DEFNODE("Sequence", "expressions", {
|
||||
},
|
||||
});
|
||||
|
||||
var AST_PropAccess = DEFNODE("PropAccess", "expression property", {
|
||||
function root_expr(prop) {
|
||||
while (prop instanceof AST_PropAccess) prop = prop.expression;
|
||||
return prop;
|
||||
}
|
||||
|
||||
var AST_PropAccess = DEFNODE("PropAccess", "expression optional property terminal", {
|
||||
$documentation: "Base class for property access expressions, i.e. `a.foo` or `a[\"foo\"]`",
|
||||
$propdoc: {
|
||||
expression: "[AST_Node] the “container” expression",
|
||||
property: "[AST_Node|string] the property to access. For AST_Dot this is always a plain string, while for AST_Sub it's an arbitrary AST_Node"
|
||||
optional: "[boolean] whether the expression is optional chaining",
|
||||
property: "[AST_Node|string] the property to access. For AST_Dot this is always a plain string, while for AST_Sub it's an arbitrary AST_Node",
|
||||
terminal: "[boolean] whether the chain has ended",
|
||||
},
|
||||
getProperty: function() {
|
||||
get_property: function() {
|
||||
var p = this.property;
|
||||
if (p instanceof AST_Constant) {
|
||||
return p.value;
|
||||
}
|
||||
if (p instanceof AST_UnaryPrefix
|
||||
&& p.operator == "void"
|
||||
&& p.expression instanceof AST_Constant) {
|
||||
return;
|
||||
}
|
||||
if (p instanceof AST_Constant) return p.value;
|
||||
if (p instanceof AST_UnaryPrefix && p.operator == "void" && p.expression instanceof AST_Constant) return;
|
||||
return p;
|
||||
},
|
||||
_validate: function() {
|
||||
@@ -1685,7 +1690,7 @@ var AST_ObjectMethod = DEFNODE("ObjectMethod", null, {
|
||||
_validate: function() {
|
||||
if (!(this.value instanceof AST_LambdaExpression)) throw new Error("value must be AST_LambdaExpression");
|
||||
if (is_arrow(this.value)) throw new Error("value cannot be AST_Arrow or AST_AsyncArrow");
|
||||
if (this.value.name != null) throw new Error("name of class method's lambda must be null");
|
||||
if (this.value.name != null) throw new Error("name of object method's lambda must be null");
|
||||
},
|
||||
}, AST_ObjectKeyVal);
|
||||
|
||||
@@ -1756,11 +1761,11 @@ var AST_SymbolLambda = DEFNODE("SymbolLambda", null, {
|
||||
|
||||
var AST_SymbolDefClass = DEFNODE("SymbolDefClass", null, {
|
||||
$documentation: "Symbol defining a class",
|
||||
}, AST_SymbolLet);
|
||||
}, AST_SymbolConst);
|
||||
|
||||
var AST_SymbolClass = DEFNODE("SymbolClass", null, {
|
||||
$documentation: "Symbol naming a class expression",
|
||||
}, AST_SymbolLet);
|
||||
}, AST_SymbolConst);
|
||||
|
||||
var AST_SymbolCatch = DEFNODE("SymbolCatch", null, {
|
||||
$documentation: "Symbol naming the exception in catch",
|
||||
@@ -1818,6 +1823,9 @@ var AST_This = DEFNODE("This", null, {
|
||||
|
||||
var AST_NewTarget = DEFNODE("NewTarget", null, {
|
||||
$documentation: "The `new.target` symbol",
|
||||
initialize: function() {
|
||||
this.name = "new.target";
|
||||
},
|
||||
_validate: function() {
|
||||
if (this.name !== "new.target") throw new Error('name must be "new.target": ' + this.name);
|
||||
},
|
||||
|
||||
3069
lib/compress.js
3069
lib/compress.js
File diff suppressed because it is too large
Load Diff
@@ -47,14 +47,12 @@ function parse_source_map(content) {
|
||||
}
|
||||
|
||||
function set_shorthand(name, options, keys) {
|
||||
if (options[name]) {
|
||||
keys.forEach(function(key) {
|
||||
if (options[key]) {
|
||||
if (typeof options[key] != "object") options[key] = {};
|
||||
if (!(name in options[key])) options[key][name] = options[name];
|
||||
}
|
||||
});
|
||||
}
|
||||
keys.forEach(function(key) {
|
||||
if (options[key]) {
|
||||
if (typeof options[key] != "object") options[key] = {};
|
||||
if (!(name in options[key])) options[key][name] = options[name];
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
function init_cache(cache) {
|
||||
@@ -75,8 +73,10 @@ function to_json(cache) {
|
||||
function minify(files, options) {
|
||||
try {
|
||||
options = defaults(options, {
|
||||
annotations: undefined,
|
||||
compress: {},
|
||||
enclose: false,
|
||||
ie: false,
|
||||
ie8: false,
|
||||
keep_fnames: false,
|
||||
mangle: {},
|
||||
@@ -94,23 +94,21 @@ function minify(files, options) {
|
||||
wrap: false,
|
||||
}, true);
|
||||
if (options.validate) AST_Node.enable_validation();
|
||||
var timings = options.timings && {
|
||||
start: Date.now()
|
||||
};
|
||||
if (options.rename === undefined) {
|
||||
options.rename = options.compress && options.mangle;
|
||||
}
|
||||
set_shorthand("ie8", options, [ "compress", "mangle", "output" ]);
|
||||
set_shorthand("keep_fnames", options, [ "compress", "mangle" ]);
|
||||
set_shorthand("toplevel", options, [ "compress", "mangle" ]);
|
||||
set_shorthand("v8", options, [ "mangle", "output" ]);
|
||||
set_shorthand("webkit", options, [ "mangle", "output" ]);
|
||||
var timings = options.timings && { start: Date.now() };
|
||||
if (options.rename === undefined) options.rename = options.compress && options.mangle;
|
||||
if (options.annotations !== undefined) set_shorthand("annotations", options, [ "compress", "output" ]);
|
||||
if (options.ie8) options.ie = options.ie || options.ie8;
|
||||
if (options.ie) set_shorthand("ie", options, [ "compress", "mangle", "output" ]);
|
||||
if (options.keep_fnames) set_shorthand("keep_fnames", options, [ "compress", "mangle" ]);
|
||||
if (options.toplevel) set_shorthand("toplevel", options, [ "compress", "mangle" ]);
|
||||
if (options.v8) set_shorthand("v8", options, [ "mangle", "output" ]);
|
||||
if (options.webkit) set_shorthand("webkit", options, [ "compress", "mangle", "output" ]);
|
||||
var quoted_props;
|
||||
if (options.mangle) {
|
||||
options.mangle = defaults(options.mangle, {
|
||||
cache: options.nameCache && (options.nameCache.vars || {}),
|
||||
eval: false,
|
||||
ie8: false,
|
||||
ie: false,
|
||||
keep_fnames: false,
|
||||
properties: false,
|
||||
reserved: [],
|
||||
@@ -208,27 +206,29 @@ function minify(files, options) {
|
||||
if (options.mangle && options.mangle.properties) mangle_properties(toplevel, options.mangle.properties);
|
||||
if (timings) timings.output = Date.now();
|
||||
var result = {};
|
||||
if (options.output.ast) {
|
||||
result.ast = toplevel;
|
||||
}
|
||||
if (!HOP(options.output, "code") || options.output.code) {
|
||||
var output = defaults(options.output, {
|
||||
ast: false,
|
||||
code: true,
|
||||
});
|
||||
if (output.ast) result.ast = toplevel;
|
||||
if (output.code) {
|
||||
if (options.sourceMap) {
|
||||
options.output.source_map = SourceMap(options.sourceMap);
|
||||
output.source_map = SourceMap(options.sourceMap);
|
||||
if (options.sourceMap.includeSources) {
|
||||
if (files instanceof AST_Toplevel) {
|
||||
throw new Error("original source content unavailable");
|
||||
} else for (var name in files) if (HOP(files, name)) {
|
||||
options.output.source_map.setSourceContent(name, files[name]);
|
||||
output.source_map.setSourceContent(name, files[name]);
|
||||
}
|
||||
}
|
||||
}
|
||||
delete options.output.ast;
|
||||
delete options.output.code;
|
||||
var stream = OutputStream(options.output);
|
||||
delete output.ast;
|
||||
delete output.code;
|
||||
var stream = OutputStream(output);
|
||||
toplevel.print(stream);
|
||||
result.code = stream.get();
|
||||
if (options.sourceMap) {
|
||||
result.map = options.output.source_map.toString();
|
||||
result.map = output.source_map.toString();
|
||||
var url = options.sourceMap.url;
|
||||
if (url) {
|
||||
result.code = result.code.replace(/\n\/\/# sourceMappingURL=\S+\s*$/, "");
|
||||
|
||||
1016
lib/mozilla-ast.js
1016
lib/mozilla-ast.js
File diff suppressed because it is too large
Load Diff
530
lib/output.js
530
lib/output.js
@@ -49,15 +49,14 @@ function is_some_comments(comment) {
|
||||
}
|
||||
|
||||
function OutputStream(options) {
|
||||
|
||||
var readonly = !options;
|
||||
options = defaults(options, {
|
||||
annotations : false,
|
||||
ascii_only : false,
|
||||
beautify : false,
|
||||
braces : false,
|
||||
comments : false,
|
||||
galio : false,
|
||||
ie8 : false,
|
||||
ie : false,
|
||||
indent_level : 4,
|
||||
indent_start : 0,
|
||||
inline_script : true,
|
||||
@@ -102,12 +101,35 @@ function OutputStream(options) {
|
||||
}
|
||||
}
|
||||
|
||||
var indentation = options.indent_start;
|
||||
var current_col = 0;
|
||||
var current_line = 1;
|
||||
var current_pos = 0;
|
||||
var OUTPUT = "";
|
||||
var indentation = options.indent_start;
|
||||
var last;
|
||||
var line_end = 0;
|
||||
var line_fixed = true;
|
||||
var mappings = options.source_map && [];
|
||||
var mapping_name;
|
||||
var mapping_token;
|
||||
var might_need_space;
|
||||
var might_need_semicolon;
|
||||
var need_newline_indented = false;
|
||||
var need_space = false;
|
||||
var newline_insert = -1;
|
||||
var stack;
|
||||
var OUTPUT;
|
||||
|
||||
function reset() {
|
||||
last = "";
|
||||
might_need_space = false;
|
||||
might_need_semicolon = false;
|
||||
stack = [];
|
||||
var str = OUTPUT;
|
||||
OUTPUT = "";
|
||||
return str;
|
||||
}
|
||||
|
||||
reset();
|
||||
var to_utf8 = options.ascii_only ? function(str, identifier) {
|
||||
if (identifier) str = str.replace(/[\ud800-\udbff][\udc00-\udfff]/g, function(ch) {
|
||||
return "\\u{" + (ch.charCodeAt(0) - 0xd7c0 << 10 | ch.charCodeAt(1) - 0xdc00).toString(16) + "}";
|
||||
@@ -140,6 +162,25 @@ function OutputStream(options) {
|
||||
return j == 0 ? str : s + str.slice(j);
|
||||
};
|
||||
|
||||
function quote_single(str) {
|
||||
return "'" + str.replace(/\x27/g, "\\'") + "'";
|
||||
}
|
||||
|
||||
function quote_double(str) {
|
||||
return '"' + str.replace(/\x22/g, '\\"') + '"';
|
||||
}
|
||||
|
||||
var quote_string = [
|
||||
null,
|
||||
quote_single,
|
||||
quote_double,
|
||||
function(str, quote) {
|
||||
return quote == "'" ? quote_single(str) : quote_double(str);
|
||||
},
|
||||
][options.quote_style] || function(str, quote, dq, sq) {
|
||||
return dq > sq ? quote_single(str) : quote_double(str);
|
||||
};
|
||||
|
||||
function make_string(str, quote) {
|
||||
var dq = 0, sq = 0;
|
||||
str = str.replace(/[\\\b\f\n\r\v\t\x22\x27\u2028\u2029\0\ufeff]/g, function(s, i) {
|
||||
@@ -152,7 +193,7 @@ function OutputStream(options) {
|
||||
case "\t": return "\\t";
|
||||
case "\b": return "\\b";
|
||||
case "\f": return "\\f";
|
||||
case "\x0B": return options.ie8 ? "\\x0B" : "\\v";
|
||||
case "\x0B": return options.ie ? "\\x0B" : "\\v";
|
||||
case "\u2028": return "\\u2028";
|
||||
case "\u2029": return "\\u2029";
|
||||
case "\ufeff": return "\\ufeff";
|
||||
@@ -161,54 +202,11 @@ function OutputStream(options) {
|
||||
}
|
||||
return s;
|
||||
});
|
||||
function quote_single() {
|
||||
return "'" + str.replace(/\x27/g, "\\'") + "'";
|
||||
}
|
||||
function quote_double() {
|
||||
return '"' + str.replace(/\x22/g, '\\"') + '"';
|
||||
}
|
||||
str = to_utf8(str);
|
||||
switch (options.quote_style) {
|
||||
case 1:
|
||||
return quote_single();
|
||||
case 2:
|
||||
return quote_double();
|
||||
case 3:
|
||||
return quote == "'" ? quote_single() : quote_double();
|
||||
default:
|
||||
return dq > sq ? quote_single() : quote_double();
|
||||
}
|
||||
}
|
||||
|
||||
function encode_string(str, quote) {
|
||||
var ret = make_string(str, quote);
|
||||
if (options.inline_script) {
|
||||
ret = ret.replace(/<\x2f(script)([>\/\t\n\f\r ])/gi, "<\\/$1$2");
|
||||
ret = ret.replace(/\x3c!--/g, "\\x3c!--");
|
||||
ret = ret.replace(/--\x3e/g, "--\\x3e");
|
||||
}
|
||||
return ret;
|
||||
}
|
||||
|
||||
function make_name(name) {
|
||||
name = name.toString();
|
||||
name = to_utf8(name, true);
|
||||
return name;
|
||||
return quote_string(to_utf8(str), quote, dq, sq);
|
||||
}
|
||||
|
||||
/* -----[ beautification/minification ]----- */
|
||||
|
||||
var has_parens = false;
|
||||
var line_end = 0;
|
||||
var line_fixed = true;
|
||||
var might_need_space = false;
|
||||
var might_need_semicolon = false;
|
||||
var need_newline_indented = false;
|
||||
var need_space = false;
|
||||
var newline_insert = -1;
|
||||
var last = "";
|
||||
var mapping_token, mapping_name, mappings = options.source_map && [];
|
||||
|
||||
var adjust_mappings = mappings ? function(line, col) {
|
||||
mappings.forEach(function(mapping) {
|
||||
mapping.line += line;
|
||||
@@ -256,8 +254,14 @@ function OutputStream(options) {
|
||||
|
||||
var requireSemicolonChars = makePredicate("( [ + * / - , .");
|
||||
|
||||
function print(str) {
|
||||
str = String(str);
|
||||
var print = options.beautify
|
||||
|| options.comments
|
||||
|| options.max_line_len
|
||||
|| options.preserve_line
|
||||
|| options.shebang
|
||||
|| !options.semicolons
|
||||
|| options.source_map
|
||||
|| options.width ? function(str) {
|
||||
var ch = str.charAt(0);
|
||||
if (need_newline_indented && ch) {
|
||||
need_newline_indented = false;
|
||||
@@ -327,7 +331,6 @@ function OutputStream(options) {
|
||||
}
|
||||
|
||||
OUTPUT += str;
|
||||
has_parens = str.slice(-1) == "(";
|
||||
current_pos += str.length;
|
||||
var a = str.split(/\r?\n/), n = a.length - 1;
|
||||
current_line += n;
|
||||
@@ -337,7 +340,30 @@ function OutputStream(options) {
|
||||
current_col = a[n].length;
|
||||
}
|
||||
last = str;
|
||||
}
|
||||
} : function(str) {
|
||||
var ch = str.charAt(0);
|
||||
var prev = last.slice(-1);
|
||||
if (might_need_semicolon) {
|
||||
might_need_semicolon = false;
|
||||
if (prev == ":" && ch == "}" || (!ch || ";}".indexOf(ch) < 0) && prev != ";") {
|
||||
OUTPUT += ";";
|
||||
might_need_space = false;
|
||||
}
|
||||
}
|
||||
if (might_need_space) {
|
||||
if (is_identifier_char(prev) && (is_identifier_char(ch) || ch == "\\")
|
||||
|| (ch == "/" && ch == prev)
|
||||
|| ((ch == "+" || ch == "-") && ch == last)
|
||||
|| str == "--" && last == "!"
|
||||
|| str == "in" && prev == "/"
|
||||
|| last == "--" && ch == ">") {
|
||||
OUTPUT += " ";
|
||||
}
|
||||
if (prev != "<" || str != "!") might_need_space = false;
|
||||
}
|
||||
OUTPUT += str;
|
||||
last = str;
|
||||
};
|
||||
|
||||
var space = options.beautify ? function() {
|
||||
print(" ");
|
||||
@@ -346,17 +372,16 @@ function OutputStream(options) {
|
||||
};
|
||||
|
||||
var indent = options.beautify ? function(half) {
|
||||
if (need_newline_indented) print("\n");
|
||||
print(repeat_string(" ", half ? indentation - (options.indent_level >> 1) : indentation));
|
||||
} : noop;
|
||||
|
||||
var with_indent = options.beautify ? function(col, cont) {
|
||||
if (col === true) col = next_indent();
|
||||
var with_indent = options.beautify ? function(cont) {
|
||||
var save_indentation = indentation;
|
||||
indentation = col;
|
||||
var ret = cont();
|
||||
indentation += options.indent_level;
|
||||
cont();
|
||||
indentation = save_indentation;
|
||||
return ret;
|
||||
} : function(col, cont) { return cont() };
|
||||
} : function(cont) { cont() };
|
||||
|
||||
var may_add_newline = options.max_line_len || options.preserve_line ? function() {
|
||||
fix_line();
|
||||
@@ -385,41 +410,28 @@ function OutputStream(options) {
|
||||
print(";");
|
||||
}
|
||||
|
||||
function next_indent() {
|
||||
return indentation + options.indent_level;
|
||||
}
|
||||
|
||||
function with_block(cont) {
|
||||
var ret;
|
||||
print("{");
|
||||
newline();
|
||||
with_indent(next_indent(), function() {
|
||||
ret = cont();
|
||||
});
|
||||
with_indent(cont);
|
||||
indent();
|
||||
print("}");
|
||||
return ret;
|
||||
}
|
||||
|
||||
function with_parens(cont) {
|
||||
print("(");
|
||||
may_add_newline();
|
||||
//XXX: still nice to have that for argument lists
|
||||
//var ret = with_indent(current_col, cont);
|
||||
var ret = cont();
|
||||
cont();
|
||||
may_add_newline();
|
||||
print(")");
|
||||
return ret;
|
||||
}
|
||||
|
||||
function with_square(cont) {
|
||||
print("[");
|
||||
may_add_newline();
|
||||
//var ret = with_indent(current_col, cont);
|
||||
var ret = cont();
|
||||
cont();
|
||||
may_add_newline();
|
||||
print("]");
|
||||
return ret;
|
||||
}
|
||||
|
||||
function comma() {
|
||||
@@ -449,31 +461,58 @@ function OutputStream(options) {
|
||||
return /^ *$/.test(OUTPUT.slice(index + 1));
|
||||
}
|
||||
|
||||
function pad_comment(token, force) {
|
||||
if (need_newline_indented) return;
|
||||
if (token.nlb && (force || !has_nlb())) {
|
||||
need_newline_indented = true;
|
||||
} else if (force) {
|
||||
need_space = true;
|
||||
}
|
||||
}
|
||||
|
||||
function print_comment(comment) {
|
||||
var value = comment.value.replace(/[@#]__PURE__/g, " ");
|
||||
if (/^\s*$/.test(value) && !/^\s*$/.test(comment.value)) return false;
|
||||
if (/comment[134]/.test(comment.type)) {
|
||||
print("//" + value);
|
||||
need_newline_indented = true;
|
||||
} else if (comment.type == "comment2") {
|
||||
print("/*" + value + "*/");
|
||||
}
|
||||
return true;
|
||||
}
|
||||
|
||||
function should_merge_comments(node, parent) {
|
||||
if (parent instanceof AST_Binary) return parent.left === node;
|
||||
if (parent.TYPE == "Call") return parent.expression === node;
|
||||
if (parent instanceof AST_Conditional) return parent.condition === node;
|
||||
if (parent instanceof AST_Dot) return parent.expression === node;
|
||||
if (parent instanceof AST_Exit) return true;
|
||||
if (parent instanceof AST_Sequence) return parent.expressions[0] === node;
|
||||
if (parent instanceof AST_Sub) return parent.expression === node;
|
||||
if (parent instanceof AST_UnaryPostfix) return true;
|
||||
if (parent instanceof AST_Yield) return true;
|
||||
}
|
||||
|
||||
function prepend_comments(node) {
|
||||
var self = this;
|
||||
var scan = node instanceof AST_Exit && node.value;
|
||||
var scan;
|
||||
if (node instanceof AST_Exit) {
|
||||
scan = node.value;
|
||||
} else if (node instanceof AST_Yield) {
|
||||
scan = node.expression;
|
||||
}
|
||||
var comments = dump(node);
|
||||
if (!comments) comments = [];
|
||||
|
||||
if (scan) {
|
||||
var tw = new TreeWalker(function(node) {
|
||||
var parent = tw.parent();
|
||||
if (parent instanceof AST_Exit
|
||||
|| parent instanceof AST_Binary && parent.left === node
|
||||
|| parent.TYPE == "Call" && parent.expression === node
|
||||
|| parent instanceof AST_Conditional && parent.condition === node
|
||||
|| parent instanceof AST_Dot && parent.expression === node
|
||||
|| parent instanceof AST_Sequence && parent.expressions[0] === node
|
||||
|| parent instanceof AST_Sub && parent.expression === node
|
||||
|| parent instanceof AST_UnaryPostfix) {
|
||||
var before = dump(node);
|
||||
if (before) comments = comments.concat(before);
|
||||
} else {
|
||||
return true;
|
||||
}
|
||||
if (!should_merge_comments(node, tw.parent())) return true;
|
||||
var before = dump(node);
|
||||
if (before) comments = comments.concat(before);
|
||||
});
|
||||
tw.push(node);
|
||||
node.value.walk(tw);
|
||||
scan.walk(tw);
|
||||
}
|
||||
|
||||
if (current_pos == 0) {
|
||||
@@ -488,35 +527,12 @@ function OutputStream(options) {
|
||||
}
|
||||
|
||||
comments = comments.filter(comment_filter, node);
|
||||
if (comments.length == 0) return;
|
||||
var last_nlb = has_nlb();
|
||||
comments.forEach(function(c, i) {
|
||||
if (!last_nlb) {
|
||||
if (c.nlb) {
|
||||
print("\n");
|
||||
indent();
|
||||
last_nlb = true;
|
||||
} else if (i > 0) {
|
||||
space();
|
||||
}
|
||||
}
|
||||
if (/comment[134]/.test(c.type)) {
|
||||
print("//" + c.value.replace(/[@#]__PURE__/g, " ") + "\n");
|
||||
indent();
|
||||
last_nlb = true;
|
||||
} else if (c.type == "comment2") {
|
||||
print("/*" + c.value.replace(/[@#]__PURE__/g, " ") + "*/");
|
||||
last_nlb = false;
|
||||
}
|
||||
var printed = false;
|
||||
comments.forEach(function(comment, index) {
|
||||
pad_comment(comment, index);
|
||||
if (print_comment(comment)) printed = true;
|
||||
});
|
||||
if (!last_nlb) {
|
||||
if (node.start.nlb) {
|
||||
print("\n");
|
||||
indent();
|
||||
} else {
|
||||
space();
|
||||
}
|
||||
}
|
||||
if (printed) pad_comment(node.start, true);
|
||||
|
||||
function dump(node) {
|
||||
var token = node.start;
|
||||
@@ -546,38 +562,21 @@ function OutputStream(options) {
|
||||
}))) return;
|
||||
comments._dumped = self;
|
||||
var insert = OUTPUT.length;
|
||||
comments.filter(comment_filter, node).forEach(function(c, i) {
|
||||
need_space = false;
|
||||
if (need_newline_indented) {
|
||||
print("\n");
|
||||
indent();
|
||||
need_newline_indented = false;
|
||||
} else if (c.nlb && (i > 0 || !has_nlb())) {
|
||||
print("\n");
|
||||
indent();
|
||||
} else if (i > 0 || !tail) {
|
||||
space();
|
||||
}
|
||||
if (/comment[134]/.test(c.type)) {
|
||||
print("//" + c.value.replace(/[@#]__PURE__/g, " "));
|
||||
need_newline_indented = true;
|
||||
} else if (c.type == "comment2") {
|
||||
print("/*" + c.value.replace(/[@#]__PURE__/g, " ") + "*/");
|
||||
need_space = true;
|
||||
}
|
||||
comments.filter(comment_filter, node).forEach(function(comment, index) {
|
||||
pad_comment(comment, index || !tail);
|
||||
print_comment(comment);
|
||||
});
|
||||
if (OUTPUT.length > insert) newline_insert = insert;
|
||||
}
|
||||
|
||||
var stack = [];
|
||||
return {
|
||||
get : get,
|
||||
toString : get,
|
||||
reset : reset,
|
||||
indent : indent,
|
||||
should_break : readonly ? noop : function() {
|
||||
return options.width && current_col - indentation >= options.width;
|
||||
},
|
||||
has_parens : function() { return has_parens },
|
||||
should_break : options.width ? function() {
|
||||
return current_col - indentation >= options.width;
|
||||
} : return_false,
|
||||
has_parens : function() { return last.slice(-1) == "(" },
|
||||
newline : newline,
|
||||
print : print,
|
||||
space : space,
|
||||
@@ -587,20 +586,21 @@ function OutputStream(options) {
|
||||
semicolon : semicolon,
|
||||
force_semicolon : force_semicolon,
|
||||
to_utf8 : to_utf8,
|
||||
print_name : function(name) { print(make_name(name)) },
|
||||
print_string : function(str, quote) { print(encode_string(str, quote)) },
|
||||
next_indent : next_indent,
|
||||
print_name : function(name) { print(to_utf8(name.toString(), true)) },
|
||||
print_string : options.inline_script ? function(str, quote) {
|
||||
str = make_string(str, quote).replace(/<\x2f(script)([>\/\t\n\f\r ])/gi, "<\\/$1$2");
|
||||
print(str.replace(/\x3c!--/g, "\\x3c!--").replace(/--\x3e/g, "--\\x3e"));
|
||||
} : function(str, quote) {
|
||||
print(make_string(str, quote));
|
||||
},
|
||||
with_indent : with_indent,
|
||||
with_block : with_block,
|
||||
with_parens : with_parens,
|
||||
with_square : with_square,
|
||||
add_mapping : add_mapping,
|
||||
option : function(opt) { return options[opt] },
|
||||
prepend_comments: readonly ? noop : prepend_comments,
|
||||
append_comments : readonly || comment_filter === return_false ? noop : append_comments,
|
||||
line : function() { return current_line },
|
||||
col : function() { return current_col },
|
||||
pos : function() { return current_pos },
|
||||
prepend_comments: options.comments || options.shebang ? prepend_comments : noop,
|
||||
append_comments : options.comments ? append_comments : noop,
|
||||
push_node : function(node) { stack.push(node) },
|
||||
pop_node : options.preserve_line ? function() {
|
||||
var node = stack.pop();
|
||||
@@ -645,10 +645,19 @@ function OutputStream(options) {
|
||||
stream.append_comments(self);
|
||||
}
|
||||
});
|
||||
var readonly = OutputStream({
|
||||
inline_script: false,
|
||||
shebang: false,
|
||||
width: false,
|
||||
});
|
||||
AST_Node.DEFMETHOD("print_to_string", function(options) {
|
||||
var s = OutputStream(options);
|
||||
this.print(s);
|
||||
return s.get();
|
||||
if (options) {
|
||||
var stream = OutputStream(options);
|
||||
this.print(stream);
|
||||
return stream.get();
|
||||
}
|
||||
this.print(readonly);
|
||||
return readonly.reset();
|
||||
});
|
||||
|
||||
/* -----[ PARENTHESES ]----- */
|
||||
@@ -700,6 +709,8 @@ function OutputStream(options) {
|
||||
// (x++)[y]
|
||||
// (typeof x).y
|
||||
if (p instanceof AST_PropAccess) return p.expression === this;
|
||||
// (~x)`foo`
|
||||
if (p instanceof AST_Template) return p.tag === this;
|
||||
}
|
||||
PARENS(AST_Await, needs_parens_unary);
|
||||
PARENS(AST_Unary, needs_parens_unary);
|
||||
@@ -741,6 +752,8 @@ function OutputStream(options) {
|
||||
|| p instanceof AST_PropAccess && p.expression === this
|
||||
// ...(foo, bar, baz)
|
||||
|| p instanceof AST_Spread
|
||||
// (foo, bar)`baz`
|
||||
|| p instanceof AST_Template && p.tag === this
|
||||
// !(foo, bar, baz)
|
||||
|| p instanceof AST_Unary
|
||||
// var a = (1, 2), b = a + a; ---> b == 4
|
||||
@@ -771,39 +784,50 @@ function OutputStream(options) {
|
||||
if (p instanceof AST_Class) return true;
|
||||
// (foo && bar)["prop"], (foo && bar).prop
|
||||
if (p instanceof AST_PropAccess) return p.expression === this;
|
||||
// (foo && bar)``
|
||||
if (p instanceof AST_Template) return p.tag === this;
|
||||
// typeof (foo && bar)
|
||||
if (p instanceof AST_Unary) return true;
|
||||
});
|
||||
|
||||
function need_chain_parens(node, parent) {
|
||||
if (!node.terminal) return false;
|
||||
if (!(parent instanceof AST_Call || parent instanceof AST_PropAccess)) return false;
|
||||
return parent.expression === node;
|
||||
}
|
||||
|
||||
PARENS(AST_PropAccess, function(output) {
|
||||
var node = this;
|
||||
var p = output.parent();
|
||||
if (p instanceof AST_New && p.expression === node) {
|
||||
// i.e. new (foo().bar)
|
||||
//
|
||||
// if there's one call into this subtree, then we need
|
||||
// parens around it too, otherwise the call will be
|
||||
// interpreted as passing the arguments to the upper New
|
||||
// expression.
|
||||
do {
|
||||
node = node.expression;
|
||||
} while (node instanceof AST_PropAccess);
|
||||
return node.TYPE == "Call";
|
||||
}
|
||||
// i.e. new (foo().bar)
|
||||
//
|
||||
// if there's one call into this subtree, then we need
|
||||
// parens around it too, otherwise the call will be
|
||||
// interpreted as passing the arguments to the upper New
|
||||
// expression.
|
||||
if (p instanceof AST_New && p.expression === node && root_expr(node).TYPE == "Call") return true;
|
||||
// (foo?.bar)()
|
||||
// (foo?.bar).baz
|
||||
// new (foo?.bar)()
|
||||
return need_chain_parens(node, p);
|
||||
});
|
||||
|
||||
PARENS(AST_Call, function(output) {
|
||||
var node = this;
|
||||
var p = output.parent();
|
||||
if (p instanceof AST_New) return p.expression === this;
|
||||
if (p instanceof AST_New) return p.expression === node;
|
||||
// https://bugs.webkit.org/show_bug.cgi?id=123506
|
||||
if (output.option("webkit")) {
|
||||
if (output.option("webkit")
|
||||
&& node.expression instanceof AST_Function
|
||||
&& p instanceof AST_PropAccess
|
||||
&& p.expression === node) {
|
||||
var g = output.parent(1);
|
||||
return this.expression instanceof AST_Function
|
||||
&& p instanceof AST_PropAccess
|
||||
&& p.expression === this
|
||||
&& g instanceof AST_Assign
|
||||
&& g.left === p;
|
||||
if (g instanceof AST_Assign && g.left === p) return true;
|
||||
}
|
||||
// (foo?.())()
|
||||
// (foo?.()).bar
|
||||
// new (foo?.())()
|
||||
return need_chain_parens(node, p);
|
||||
});
|
||||
|
||||
PARENS(AST_New, function(output) {
|
||||
@@ -929,7 +953,7 @@ function OutputStream(options) {
|
||||
});
|
||||
function print_braced_empty(self, output) {
|
||||
output.print("{");
|
||||
output.with_indent(output.next_indent(), function() {
|
||||
output.with_indent(function() {
|
||||
output.append_comments(self, true);
|
||||
});
|
||||
output.print("}");
|
||||
@@ -950,7 +974,6 @@ function OutputStream(options) {
|
||||
DEFPRINT(AST_Do, function(output) {
|
||||
var self = this;
|
||||
output.print("do");
|
||||
output.space();
|
||||
make_block(self.body, output);
|
||||
output.space();
|
||||
output.print("while");
|
||||
@@ -967,7 +990,6 @@ function OutputStream(options) {
|
||||
output.with_parens(function() {
|
||||
self.condition.print(output);
|
||||
});
|
||||
output.space();
|
||||
force_statement(self.body, output);
|
||||
});
|
||||
DEFPRINT(AST_For, function(output) {
|
||||
@@ -997,7 +1019,6 @@ function OutputStream(options) {
|
||||
self.step.print(output);
|
||||
}
|
||||
});
|
||||
output.space();
|
||||
force_statement(self.body, output);
|
||||
});
|
||||
function print_for_enum(prefix, infix) {
|
||||
@@ -1012,7 +1033,6 @@ function OutputStream(options) {
|
||||
output.space();
|
||||
self.object.print(output);
|
||||
});
|
||||
output.space();
|
||||
force_statement(self.body, output);
|
||||
};
|
||||
}
|
||||
@@ -1026,7 +1046,6 @@ function OutputStream(options) {
|
||||
output.with_parens(function() {
|
||||
self.expression.print(output);
|
||||
});
|
||||
output.space();
|
||||
force_statement(self.body, output);
|
||||
});
|
||||
DEFPRINT(AST_ExportDeclaration, function(output) {
|
||||
@@ -1265,7 +1284,7 @@ function OutputStream(options) {
|
||||
function make_then(self, output) {
|
||||
var b = self.body;
|
||||
if (output.option("braces") && !(b instanceof AST_Const || b instanceof AST_Let)
|
||||
|| output.option("ie8") && b instanceof AST_Do)
|
||||
|| output.option("ie") && b instanceof AST_Do)
|
||||
return make_block(b, output);
|
||||
// The squeezer replaces "block"-s that contain only a single
|
||||
// statement with the statement itself; technically, the AST
|
||||
@@ -1295,16 +1314,16 @@ function OutputStream(options) {
|
||||
output.with_parens(function() {
|
||||
self.condition.print(output);
|
||||
});
|
||||
output.space();
|
||||
if (self.alternative) {
|
||||
make_then(self, output);
|
||||
output.space();
|
||||
output.print("else");
|
||||
output.space();
|
||||
if (self.alternative instanceof AST_If)
|
||||
if (self.alternative instanceof AST_If) {
|
||||
output.space();
|
||||
self.alternative.print(output);
|
||||
else
|
||||
} else {
|
||||
force_statement(self.alternative, output);
|
||||
}
|
||||
} else {
|
||||
force_statement(self.body, output);
|
||||
}
|
||||
@@ -1436,6 +1455,17 @@ function OutputStream(options) {
|
||||
});
|
||||
|
||||
/* -----[ other expressions ]----- */
|
||||
function print_annotation(self, output) {
|
||||
if (!output.option("annotations")) return;
|
||||
if (!self.pure) return;
|
||||
var level = 0, parent = self, node;
|
||||
do {
|
||||
node = parent;
|
||||
parent = output.parent(level++);
|
||||
if (parent instanceof AST_Call && parent.expression === node) return;
|
||||
} while (parent instanceof AST_PropAccess && parent.expression === node);
|
||||
output.print(typeof self.pure == "string" ? "/*" + self.pure + "*/" : "/*@__PURE__*/");
|
||||
}
|
||||
function print_call_args(self, output) {
|
||||
if (self.expression instanceof AST_Call || self.expression instanceof AST_Lambda) {
|
||||
output.add_mapping(self.start);
|
||||
@@ -1448,11 +1478,15 @@ function OutputStream(options) {
|
||||
});
|
||||
}
|
||||
DEFPRINT(AST_Call, function(output) {
|
||||
this.expression.print(output);
|
||||
print_call_args(this, output);
|
||||
var self = this;
|
||||
print_annotation(self, output);
|
||||
self.expression.print(output);
|
||||
if (self.optional) output.print("?.");
|
||||
print_call_args(self, output);
|
||||
});
|
||||
DEFPRINT(AST_New, function(output) {
|
||||
var self = this;
|
||||
print_annotation(self, output);
|
||||
output.print("new");
|
||||
output.space();
|
||||
self.expression.print(output);
|
||||
@@ -1475,27 +1509,24 @@ function OutputStream(options) {
|
||||
var expr = self.expression;
|
||||
expr.print(output);
|
||||
var prop = self.property;
|
||||
if (output.option("ie8") && RESERVED_WORDS[prop]) {
|
||||
output.print("[");
|
||||
if (output.option("ie") && RESERVED_WORDS[prop]) {
|
||||
output.print(self.optional ? "?.[" : "[");
|
||||
output.add_mapping(self.end);
|
||||
output.print_string(prop);
|
||||
output.print("]");
|
||||
} else {
|
||||
if (expr instanceof AST_Number && expr.value >= 0) {
|
||||
if (!/[xa-f.)]/i.test(output.last())) {
|
||||
output.print(".");
|
||||
}
|
||||
}
|
||||
output.print(".");
|
||||
if (expr instanceof AST_Number && !/[ex.)]/i.test(output.last())) output.print(".");
|
||||
output.print(self.optional ? "?." : ".");
|
||||
// the name after dot would be mapped about here.
|
||||
output.add_mapping(self.end);
|
||||
output.print_name(prop);
|
||||
}
|
||||
});
|
||||
DEFPRINT(AST_Sub, function(output) {
|
||||
this.expression.print(output);
|
||||
output.print("[");
|
||||
this.property.print(output);
|
||||
var self = this;
|
||||
self.expression.print(output);
|
||||
output.print(self.optional ? "?.[" : "[");
|
||||
self.property.print(output);
|
||||
output.print("]");
|
||||
});
|
||||
DEFPRINT(AST_Spread, function(output) {
|
||||
@@ -1588,7 +1619,26 @@ function OutputStream(options) {
|
||||
output.space();
|
||||
} : noop);
|
||||
});
|
||||
DEFPRINT(AST_DestructuredKeyVal, print_key_value);
|
||||
DEFPRINT(AST_DestructuredKeyVal, function(output) {
|
||||
var self = this;
|
||||
var key = print_property_key(self, output);
|
||||
var value = self.value;
|
||||
if (key) {
|
||||
if (value instanceof AST_DefaultValue) {
|
||||
if (value.name instanceof AST_Symbol && key == get_symbol_name(value.name)) {
|
||||
output.space();
|
||||
output.print("=");
|
||||
output.space();
|
||||
value.value.print(output);
|
||||
return;
|
||||
}
|
||||
} else if (value instanceof AST_Symbol) {
|
||||
if (key == get_symbol_name(value)) return;
|
||||
}
|
||||
}
|
||||
output.colon();
|
||||
value.print(output);
|
||||
});
|
||||
DEFPRINT(AST_DestructuredObject, function(output) {
|
||||
var props = this.properties, len = props.length, rest = this.rest;
|
||||
if (len || rest) output.with_block(function() {
|
||||
@@ -1634,37 +1684,29 @@ function OutputStream(options) {
|
||||
|
||||
function print_property_key(self, output) {
|
||||
var key = self.key;
|
||||
if (key instanceof AST_Node) {
|
||||
output.with_square(function() {
|
||||
key.print(output);
|
||||
});
|
||||
} else if (output.option("quote_keys")) {
|
||||
output.print_string(key);
|
||||
if (key instanceof AST_Node) return output.with_square(function() {
|
||||
key.print(output);
|
||||
});
|
||||
var quote = self.start && self.start.quote;
|
||||
if (output.option("quote_keys") || quote && output.option("keep_quoted_props")) {
|
||||
output.print_string(key, quote);
|
||||
} else if ("" + +key == key && key >= 0) {
|
||||
output.print(make_num(key));
|
||||
} else if (self.private) {
|
||||
output.print_name(key);
|
||||
} else if (RESERVED_WORDS[key] ? !output.option("ie") : is_identifier_string(key)) {
|
||||
output.print_name(key);
|
||||
return key;
|
||||
} else {
|
||||
var quote = self.start && self.start.quote;
|
||||
if (self.private) {
|
||||
output.print_name(key);
|
||||
} else if (RESERVED_WORDS[key] ? !output.option("ie8") : is_identifier_string(key)) {
|
||||
if (quote && output.option("keep_quoted_props")) {
|
||||
output.print_string(key, quote);
|
||||
} else {
|
||||
output.print_name(key);
|
||||
}
|
||||
} else {
|
||||
output.print_string(key, quote);
|
||||
}
|
||||
output.print_string(key, quote);
|
||||
}
|
||||
}
|
||||
|
||||
function print_key_value(output) {
|
||||
DEFPRINT(AST_ObjectKeyVal, function(output) {
|
||||
var self = this;
|
||||
print_property_key(self, output);
|
||||
output.colon();
|
||||
self.value.print(output);
|
||||
}
|
||||
DEFPRINT(AST_ObjectKeyVal, print_key_value);
|
||||
});
|
||||
DEFPRINT(AST_ObjectMethod, function(output) {
|
||||
print_method(this, output);
|
||||
});
|
||||
@@ -1683,32 +1725,36 @@ function OutputStream(options) {
|
||||
}
|
||||
DEFPRINT(AST_ObjectGetter, print_accessor("get"));
|
||||
DEFPRINT(AST_ObjectSetter, print_accessor("set"));
|
||||
function print_symbol(self, output) {
|
||||
var def = self.definition();
|
||||
output.print_name(def && def.mangled_name || self.name);
|
||||
function get_symbol_name(sym) {
|
||||
var def = sym.definition();
|
||||
return def && def.mangled_name || sym.name;
|
||||
}
|
||||
DEFPRINT(AST_Symbol, function(output) {
|
||||
print_symbol(this, output);
|
||||
output.print_name(get_symbol_name(this));
|
||||
});
|
||||
DEFPRINT(AST_SymbolExport, function(output) {
|
||||
var self = this;
|
||||
print_symbol(self, output);
|
||||
if (self.alias) {
|
||||
var name = get_symbol_name(self);
|
||||
output.print_name(name);
|
||||
var alias = self.alias;
|
||||
if (alias != name) {
|
||||
output.space();
|
||||
output.print("as");
|
||||
output.space();
|
||||
output.print_name(self.alias);
|
||||
output.print_name(alias);
|
||||
}
|
||||
});
|
||||
DEFPRINT(AST_SymbolImport, function(output) {
|
||||
var self = this;
|
||||
if (self.key) {
|
||||
output.print_name(self.key);
|
||||
var name = get_symbol_name(self);
|
||||
var key = self.key;
|
||||
if (key && key != name) {
|
||||
output.print_name(key);
|
||||
output.space();
|
||||
output.print("as");
|
||||
output.space();
|
||||
}
|
||||
print_symbol(self, output);
|
||||
output.print_name(name);
|
||||
});
|
||||
DEFPRINT(AST_Hole, noop);
|
||||
DEFPRINT(AST_Template, function(output) {
|
||||
@@ -1725,7 +1771,7 @@ function OutputStream(options) {
|
||||
output.print("`");
|
||||
});
|
||||
DEFPRINT(AST_Constant, function(output) {
|
||||
output.print(this.value);
|
||||
output.print("" + this.value);
|
||||
});
|
||||
DEFPRINT(AST_String, function(output) {
|
||||
output.print_string(this.value, this.quote);
|
||||
@@ -1781,9 +1827,10 @@ function OutputStream(options) {
|
||||
function force_statement(stat, output) {
|
||||
if (output.option("braces") && !(stat instanceof AST_Const || stat instanceof AST_Let)) {
|
||||
make_block(stat, output);
|
||||
} else if (!stat || stat instanceof AST_EmptyStatement) {
|
||||
} else if (stat instanceof AST_EmptyStatement) {
|
||||
output.force_semicolon();
|
||||
} else {
|
||||
output.space();
|
||||
stat.print(output);
|
||||
}
|
||||
}
|
||||
@@ -1832,11 +1879,12 @@ function OutputStream(options) {
|
||||
}
|
||||
|
||||
function make_block(stmt, output) {
|
||||
if (!stmt || stmt instanceof AST_EmptyStatement)
|
||||
output.print("{}");
|
||||
else if (stmt instanceof AST_BlockStatement)
|
||||
output.space();
|
||||
if (stmt instanceof AST_EmptyStatement) {
|
||||
print_braced_empty(stmt, output);
|
||||
} else if (stmt instanceof AST_BlockStatement) {
|
||||
stmt.print(output);
|
||||
else output.with_block(function() {
|
||||
} else output.with_block(function() {
|
||||
output.indent();
|
||||
stmt.print(output);
|
||||
output.newline();
|
||||
@@ -1883,7 +1931,11 @@ function OutputStream(options) {
|
||||
output.add_mapping(this.start);
|
||||
});
|
||||
|
||||
DEFMAP([ AST_DestructuredKeyVal, AST_ObjectProperty ], function(output) {
|
||||
DEFMAP([
|
||||
AST_ClassProperty,
|
||||
AST_DestructuredKeyVal,
|
||||
AST_ObjectProperty,
|
||||
], function(output) {
|
||||
if (typeof this.key == "string") output.add_mapping(this.start, this.key);
|
||||
});
|
||||
})();
|
||||
|
||||
290
lib/parse.js
290
lib/parse.js
@@ -100,15 +100,19 @@ var OPERATORS = makePredicate([
|
||||
"/=",
|
||||
"*=",
|
||||
"%=",
|
||||
"**=",
|
||||
">>=",
|
||||
"<<=",
|
||||
">>>=",
|
||||
"&=",
|
||||
"|=",
|
||||
"^=",
|
||||
"&=",
|
||||
"&&",
|
||||
"||",
|
||||
"??",
|
||||
"&&=",
|
||||
"||=",
|
||||
"??=",
|
||||
]);
|
||||
|
||||
var NEWLINE_CHARS = "\n\r\u2028\u2029";
|
||||
@@ -120,7 +124,7 @@ var PUNC_AFTER_EXPRESSION = PUNC_SEPARATORS + PUNC_CLOSERS;
|
||||
var PUNC_BEFORE_EXPRESSION = PUNC_OPENERS + PUNC_SEPARATORS;
|
||||
var PUNC_CHARS = PUNC_BEFORE_EXPRESSION + "`" + PUNC_CLOSERS;
|
||||
var WHITESPACE_CHARS = NEWLINE_CHARS + " \u00a0\t\f\u000b\u200b\u2000\u2001\u2002\u2003\u2004\u2005\u2006\u2007\u2008\u2009\u200a\u202f\u205f\u3000\uFEFF";
|
||||
var NON_IDENTIFIER_CHARS = makePredicate(characters("./'\"" + OPERATOR_CHARS + PUNC_CHARS + WHITESPACE_CHARS));
|
||||
var NON_IDENTIFIER_CHARS = makePredicate(characters("./'\"#" + OPERATOR_CHARS + PUNC_CHARS + WHITESPACE_CHARS));
|
||||
|
||||
NEWLINE_CHARS = makePredicate(characters(NEWLINE_CHARS));
|
||||
OPERATOR_CHARS = makePredicate(characters(OPERATOR_CHARS));
|
||||
@@ -160,10 +164,10 @@ function decode_escape_sequence(seq) {
|
||||
case "t": return "\t";
|
||||
case "u":
|
||||
var code;
|
||||
if (seq.length == 5) {
|
||||
code = seq.slice(1);
|
||||
} else if (seq[1] == "{" && seq.slice(-1) == "}") {
|
||||
if (seq[1] == "{" && seq.slice(-1) == "}") {
|
||||
code = seq.slice(2, -1);
|
||||
} else if (seq.length == 5) {
|
||||
code = seq.slice(1);
|
||||
} else {
|
||||
return;
|
||||
}
|
||||
@@ -347,7 +351,7 @@ function tokenizer($TEXT, filename, html5_comments, shebang) {
|
||||
|
||||
function read_while(pred) {
|
||||
var ret = "", ch;
|
||||
while ((ch = peek()) && pred(ch)) ret += next();
|
||||
while ((ch = peek()) && pred(ch, ret)) ret += next();
|
||||
return ret;
|
||||
}
|
||||
|
||||
@@ -355,24 +359,27 @@ function tokenizer($TEXT, filename, html5_comments, shebang) {
|
||||
js_error(err, filename, S.tokline, S.tokcol, S.tokpos);
|
||||
}
|
||||
|
||||
function is_octal(num) {
|
||||
return /^0[0-7_]+$/.test(num);
|
||||
}
|
||||
|
||||
function read_num(prefix) {
|
||||
var has_e = false, after_e = false, has_x = false, has_dot = prefix == ".";
|
||||
var num = read_while(function(ch) {
|
||||
var code = ch.charCodeAt(0);
|
||||
switch (code) {
|
||||
case 120: case 88: // xX
|
||||
var num = read_while(function(ch, str) {
|
||||
switch (ch) {
|
||||
case "x": case "X":
|
||||
return has_x ? false : (has_x = true);
|
||||
case 101: case 69: // eE
|
||||
case "e": case "E":
|
||||
return has_x ? true : has_e ? false : (has_e = after_e = true);
|
||||
case 43: case 45: // +-
|
||||
case "+": case "-":
|
||||
return after_e;
|
||||
case (after_e = false, 46): // .
|
||||
return (!has_dot && !has_x && !has_e) ? (has_dot = true) : false;
|
||||
case (after_e = false, "."):
|
||||
return has_dot || has_e || has_x || is_octal(str) ? false : (has_dot = true);
|
||||
}
|
||||
return is_digit(code) || /[_0-9a-fo]/i.test(ch);
|
||||
return /[_0-9a-dfo]/i.test(ch);
|
||||
});
|
||||
if (prefix) num = prefix + num;
|
||||
if (/^0[0-7_]+$/.test(num)) {
|
||||
if (is_octal(num)) {
|
||||
if (next_token.has_directive("use strict")) parse_error("Legacy octal literals are not allowed in strict mode");
|
||||
} else {
|
||||
num = num.replace(has_x ? /([1-9a-f]|.0)_(?=[0-9a-f])/gi : /([1-9]|.0)_(?=[0-9])/gi, "$1");
|
||||
@@ -461,7 +468,7 @@ function tokenizer($TEXT, filename, html5_comments, shebang) {
|
||||
});
|
||||
|
||||
function read_name() {
|
||||
var backslash = false, name = "", ch, escaped = false, hex;
|
||||
var backslash = false, ch, escaped = false, name = peek() == "#" ? next() : "";
|
||||
while (ch = peek()) {
|
||||
if (!backslash) {
|
||||
if (ch == "\\") escaped = backslash = true, next();
|
||||
@@ -476,7 +483,7 @@ function tokenizer($TEXT, filename, html5_comments, shebang) {
|
||||
}
|
||||
}
|
||||
if (KEYWORDS[name] && escaped) {
|
||||
hex = name.charCodeAt(0).toString(16).toUpperCase();
|
||||
var hex = name.charCodeAt(0).toString(16).toUpperCase();
|
||||
name = "\\u" + "0000".substr(hex.length) + hex + name.slice(1);
|
||||
}
|
||||
return name;
|
||||
@@ -611,7 +618,7 @@ function tokenizer($TEXT, filename, html5_comments, shebang) {
|
||||
if (PUNC_CHARS[ch]) return token("punc", next());
|
||||
if (looking_at("=>")) return token("punc", next() + next());
|
||||
if (OPERATOR_CHARS[ch]) return read_operator();
|
||||
if (code == 92 || !NON_IDENTIFIER_CHARS[ch]) return read_word();
|
||||
if (code == 35 || code == 92 || !NON_IDENTIFIER_CHARS[ch]) return read_word();
|
||||
break;
|
||||
}
|
||||
parse_error("Unexpected character '" + ch + "'");
|
||||
@@ -652,7 +659,7 @@ var UNARY_PREFIX = makePredicate("typeof void delete -- ++ ! ~ - +");
|
||||
|
||||
var UNARY_POSTFIX = makePredicate("-- ++");
|
||||
|
||||
var ASSIGNMENT = makePredicate("= += -= /= *= %= >>= <<= >>>= |= ^= &=");
|
||||
var ASSIGNMENT = makePredicate("= += -= /= *= %= **= >>= <<= >>>= &= |= ^= &&= ||= ??=");
|
||||
|
||||
var PRECEDENCE = function(a, ret) {
|
||||
for (var i = 0; i < a.length;) {
|
||||
@@ -850,7 +857,8 @@ function parse($TEXT, options) {
|
||||
next();
|
||||
return export_();
|
||||
case "import":
|
||||
if (!is_token(peek(), "punc", "(")) {
|
||||
var token = peek();
|
||||
if (!(token.type == "punc" && /^[(.]$/.test(token.value))) {
|
||||
next();
|
||||
return import_();
|
||||
}
|
||||
@@ -1181,7 +1189,7 @@ function parse($TEXT, options) {
|
||||
var await = is("name", "await") && next();
|
||||
expect("(");
|
||||
var init = null;
|
||||
if (!is("punc", ";")) {
|
||||
if (await || !is("punc", ";")) {
|
||||
init = is("keyword", "const")
|
||||
? (next(), const_(true))
|
||||
: is("keyword", "let")
|
||||
@@ -1232,6 +1240,7 @@ function parse($TEXT, options) {
|
||||
}
|
||||
|
||||
function for_enum(ctor, init) {
|
||||
handle_regexp();
|
||||
var obj = expression();
|
||||
expect(")");
|
||||
return new ctor({
|
||||
@@ -1325,21 +1334,19 @@ function parse($TEXT, options) {
|
||||
if (is("punc", "{")) {
|
||||
body = block_();
|
||||
value = null;
|
||||
if (S.input.has_directive("use strict")) {
|
||||
argnames.forEach(strict_verify_symbol);
|
||||
}
|
||||
} else {
|
||||
body = [];
|
||||
handle_regexp();
|
||||
value = maybe_assign();
|
||||
}
|
||||
var is_strict = S.input.has_directive("use strict");
|
||||
S.input.pop_directives_stack();
|
||||
--S.in_function;
|
||||
S.in_loop = loop;
|
||||
S.labels = labels;
|
||||
S.in_generator = was_gen;
|
||||
S.in_async = was_async;
|
||||
return new (async ? AST_AsyncArrow : AST_Arrow)({
|
||||
var node = new (async ? AST_AsyncArrow : AST_Arrow)({
|
||||
start: start,
|
||||
argnames: argnames,
|
||||
rest: rest,
|
||||
@@ -1347,6 +1354,8 @@ function parse($TEXT, options) {
|
||||
value: value,
|
||||
end: prev(),
|
||||
});
|
||||
if (is_strict) node.each_argname(strict_verify_symbol);
|
||||
return node;
|
||||
}
|
||||
|
||||
var function_ = function(ctor) {
|
||||
@@ -1379,23 +1388,24 @@ function parse($TEXT, options) {
|
||||
S.in_loop = 0;
|
||||
S.labels = [];
|
||||
var body = block_();
|
||||
if (S.input.has_directive("use strict")) {
|
||||
if (name) strict_verify_symbol(name);
|
||||
argnames.forEach(strict_verify_symbol);
|
||||
if (argnames.rest) strict_verify_symbol(argnames.rest);
|
||||
}
|
||||
var is_strict = S.input.has_directive("use strict");
|
||||
S.input.pop_directives_stack();
|
||||
--S.in_function;
|
||||
S.in_loop = loop;
|
||||
S.labels = labels;
|
||||
S.in_generator = was_gen;
|
||||
S.in_async = was_async;
|
||||
return new ctor({
|
||||
var node = new ctor({
|
||||
name: name,
|
||||
argnames: argnames,
|
||||
rest: argnames.rest || null,
|
||||
body: body
|
||||
});
|
||||
if (is_strict) {
|
||||
if (name) strict_verify_symbol(name);
|
||||
node.each_argname(strict_verify_symbol);
|
||||
}
|
||||
return node;
|
||||
};
|
||||
|
||||
function if_() {
|
||||
@@ -1487,6 +1497,7 @@ function parse($TEXT, options) {
|
||||
body.start = start;
|
||||
body.end = prev();
|
||||
} else {
|
||||
handle_regexp();
|
||||
body = expression();
|
||||
semicolon();
|
||||
}
|
||||
@@ -1736,67 +1747,66 @@ function parse($TEXT, options) {
|
||||
var new_ = function(allow_calls) {
|
||||
var start = S.token;
|
||||
expect_token("operator", "new");
|
||||
var call;
|
||||
if (is("punc", ".") && is_token(peek(), "name", "target")) {
|
||||
next();
|
||||
next();
|
||||
return new AST_NewTarget({
|
||||
name: "new.target",
|
||||
start: start,
|
||||
end: prev(),
|
||||
});
|
||||
}
|
||||
var newexp = expr_atom(false), args;
|
||||
if (is("punc", "(")) {
|
||||
next();
|
||||
args = expr_list(")", !options.strict);
|
||||
call = new AST_NewTarget();
|
||||
} else {
|
||||
args = [];
|
||||
var exp = expr_atom(false), args;
|
||||
if (is("punc", "(")) {
|
||||
next();
|
||||
args = expr_list(")", !options.strict);
|
||||
} else {
|
||||
args = [];
|
||||
}
|
||||
call = new AST_New({ expression: exp, args: args });
|
||||
}
|
||||
var call = new AST_New({
|
||||
start : start,
|
||||
expression : newexp,
|
||||
args : args,
|
||||
end : prev()
|
||||
});
|
||||
mark_pure(call);
|
||||
call.start = start;
|
||||
call.end = prev();
|
||||
return subscripts(call, allow_calls);
|
||||
};
|
||||
|
||||
function as_atom_node() {
|
||||
var tok = S.token, ret;
|
||||
var ret, tok = S.token, value = tok.value;
|
||||
switch (tok.type) {
|
||||
case "num":
|
||||
ret = new AST_Number({ start: tok, end: tok, value: tok.value });
|
||||
break;
|
||||
case "bigint":
|
||||
ret = new AST_BigInt({ start: tok, end: tok, value: tok.value });
|
||||
break;
|
||||
case "string":
|
||||
ret = new AST_String({
|
||||
start : tok,
|
||||
end : tok,
|
||||
value : tok.value,
|
||||
quote : tok.quote
|
||||
});
|
||||
break;
|
||||
case "regexp":
|
||||
ret = new AST_RegExp({ start: tok, end: tok, value: tok.value });
|
||||
break;
|
||||
case "atom":
|
||||
switch (tok.value) {
|
||||
case "false":
|
||||
ret = new AST_False({ start: tok, end: tok });
|
||||
break;
|
||||
case "true":
|
||||
ret = new AST_True({ start: tok, end: tok });
|
||||
break;
|
||||
case "null":
|
||||
ret = new AST_Null({ start: tok, end: tok });
|
||||
break;
|
||||
if (isFinite(value)) {
|
||||
ret = new AST_Number({ value: value });
|
||||
} else {
|
||||
ret = new AST_Infinity();
|
||||
if (value < 0) ret = new AST_UnaryPrefix({ operator: "-", expression: ret });
|
||||
}
|
||||
break;
|
||||
case "bigint":
|
||||
ret = new AST_BigInt({ value: value });
|
||||
break;
|
||||
case "string":
|
||||
ret = new AST_String({ value : value, quote : tok.quote });
|
||||
break;
|
||||
case "regexp":
|
||||
ret = new AST_RegExp({ value: value });
|
||||
break;
|
||||
case "atom":
|
||||
switch (value) {
|
||||
case "false":
|
||||
ret = new AST_False();
|
||||
break;
|
||||
case "true":
|
||||
ret = new AST_True();
|
||||
break;
|
||||
case "null":
|
||||
ret = new AST_Null();
|
||||
break;
|
||||
default:
|
||||
unexpected();
|
||||
}
|
||||
break;
|
||||
default:
|
||||
unexpected();
|
||||
}
|
||||
next();
|
||||
ret.start = ret.end = tok;
|
||||
return ret;
|
||||
}
|
||||
|
||||
@@ -1808,10 +1818,7 @@ function parse($TEXT, options) {
|
||||
if (is("punc")) {
|
||||
switch (start.value) {
|
||||
case "`":
|
||||
var tmpl = template(null);
|
||||
tmpl.start = start;
|
||||
tmpl.end = prev();
|
||||
return subscripts(tmpl, allow_calls);
|
||||
return subscripts(template(null), allow_calls);
|
||||
case "(":
|
||||
next();
|
||||
if (is("punc", ")")) {
|
||||
@@ -1836,11 +1843,13 @@ function parse($TEXT, options) {
|
||||
expect(")");
|
||||
var end = prev();
|
||||
end.comments_before = ex.end.comments_before;
|
||||
[].push.apply(ex.end.comments_after, end.comments_after);
|
||||
end.comments_after.forEach(function(comment) {
|
||||
ex.end.comments_after.push(comment);
|
||||
if (comment.nlb) S.token.nlb = true;
|
||||
});
|
||||
end.comments_after.length = 0;
|
||||
end.comments_after = ex.end.comments_after;
|
||||
ex.end = end;
|
||||
if (ex instanceof AST_Call) mark_pure(ex);
|
||||
if (is("punc", "=>")) return arrow(ex instanceof AST_Sequence ? ex.expressions : [ ex ], start);
|
||||
return subscripts(ex, allow_calls);
|
||||
case "[":
|
||||
@@ -2217,20 +2226,8 @@ function parse($TEXT, options) {
|
||||
});
|
||||
}
|
||||
|
||||
function mark_pure(call) {
|
||||
var start = call.start;
|
||||
var comments = start.comments_before;
|
||||
var i = HOP(start, "comments_before_length") ? start.comments_before_length : comments.length;
|
||||
while (--i >= 0) {
|
||||
var comment = comments[i];
|
||||
if (/[@#]__PURE__/.test(comment.value)) {
|
||||
call.pure = comment;
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
function template(tag) {
|
||||
var start = tag ? tag.start : S.token;
|
||||
var read = S.input.context().read_template;
|
||||
var strings = [];
|
||||
var expressions = [];
|
||||
@@ -2241,53 +2238,74 @@ function parse($TEXT, options) {
|
||||
}
|
||||
next();
|
||||
return new AST_Template({
|
||||
start: start,
|
||||
expressions: expressions,
|
||||
strings: strings,
|
||||
tag: tag,
|
||||
end: prev(),
|
||||
});
|
||||
}
|
||||
|
||||
var subscripts = function(expr, allow_calls) {
|
||||
function subscripts(expr, allow_calls) {
|
||||
var start = expr.start;
|
||||
if (is("punc", ".")) {
|
||||
next();
|
||||
return subscripts(new AST_Dot({
|
||||
start : start,
|
||||
expression : expr,
|
||||
property : as_name(),
|
||||
end : prev()
|
||||
}), allow_calls);
|
||||
var optional = null;
|
||||
while (true) {
|
||||
if (is("operator", "?") && is_token(peek(), "punc", ".")) {
|
||||
next();
|
||||
next();
|
||||
optional = expr;
|
||||
}
|
||||
if (is("punc", "[")) {
|
||||
next();
|
||||
var prop = expression();
|
||||
expect("]");
|
||||
expr = new AST_Sub({
|
||||
start: start,
|
||||
optional: optional === expr,
|
||||
expression: expr,
|
||||
property: prop,
|
||||
end: prev(),
|
||||
});
|
||||
} else if (allow_calls && is("punc", "(")) {
|
||||
next();
|
||||
expr = new AST_Call({
|
||||
start: start,
|
||||
optional: optional === expr,
|
||||
expression: expr,
|
||||
args: expr_list(")", !options.strict),
|
||||
end: prev(),
|
||||
});
|
||||
} else if (optional === expr || is("punc", ".")) {
|
||||
if (optional !== expr) next();
|
||||
expr = new AST_Dot({
|
||||
start: start,
|
||||
optional: optional === expr,
|
||||
expression: expr,
|
||||
property: as_name(),
|
||||
end: prev(),
|
||||
});
|
||||
} else if (is("punc", "`")) {
|
||||
if (optional) croak("Invalid template on optional chain");
|
||||
expr = template(expr);
|
||||
} else {
|
||||
break;
|
||||
}
|
||||
}
|
||||
if (is("punc", "[")) {
|
||||
next();
|
||||
var prop = expression();
|
||||
expect("]");
|
||||
return subscripts(new AST_Sub({
|
||||
start : start,
|
||||
expression : expr,
|
||||
property : prop,
|
||||
end : prev()
|
||||
}), allow_calls);
|
||||
}
|
||||
if (allow_calls && is("punc", "(")) {
|
||||
next();
|
||||
var call = new AST_Call({
|
||||
start : start,
|
||||
expression : expr,
|
||||
args : expr_list(")", !options.strict),
|
||||
end : prev()
|
||||
});
|
||||
mark_pure(call);
|
||||
return subscripts(call, true);
|
||||
}
|
||||
if (is("punc", "`")) {
|
||||
var tmpl = template(expr);
|
||||
tmpl.start = expr.start;
|
||||
tmpl.end = prev();
|
||||
return subscripts(tmpl, allow_calls);
|
||||
if (optional) expr.terminal = true;
|
||||
if (expr instanceof AST_Call && !expr.pure) {
|
||||
var start = expr.start;
|
||||
var comments = start.comments_before;
|
||||
var i = HOP(start, "comments_before_length") ? start.comments_before_length : comments.length;
|
||||
while (--i >= 0) {
|
||||
var match = /[@#]__PURE__/.exec(comments[i].value);
|
||||
if (match) {
|
||||
expr.pure = match[0];
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
return expr;
|
||||
};
|
||||
}
|
||||
|
||||
function maybe_unary(no_in) {
|
||||
var start = S.token;
|
||||
@@ -2396,7 +2414,7 @@ function parse($TEXT, options) {
|
||||
};
|
||||
|
||||
function is_assignable(expr) {
|
||||
return expr instanceof AST_PropAccess || expr instanceof AST_SymbolRef;
|
||||
return expr instanceof AST_PropAccess && !expr.optional || expr instanceof AST_SymbolRef;
|
||||
}
|
||||
|
||||
function to_destructured(node) {
|
||||
@@ -2518,7 +2536,7 @@ function parse($TEXT, options) {
|
||||
while (!is("eof"))
|
||||
body.push(statement());
|
||||
S.input.pop_directives_stack();
|
||||
var end = prev();
|
||||
var end = prev() || start;
|
||||
var toplevel = options.toplevel;
|
||||
if (toplevel) {
|
||||
toplevel.body = toplevel.body.concat(body);
|
||||
|
||||
@@ -81,7 +81,9 @@ var builtins = function() {
|
||||
|
||||
function reserve_quoted_keys(ast, reserved) {
|
||||
ast.walk(new TreeWalker(function(node) {
|
||||
if (node instanceof AST_ObjectProperty) {
|
||||
if (node instanceof AST_ClassProperty) {
|
||||
if (node.start && node.start.quote) add(node.key);
|
||||
} else if (node instanceof AST_ObjectProperty) {
|
||||
if (node.start && node.start.quote) add(node.key);
|
||||
} else if (node instanceof AST_Sub) {
|
||||
addStrings(node.property, add);
|
||||
@@ -163,6 +165,8 @@ function mangle_properties(ast, options) {
|
||||
addStrings(node.args[0], add);
|
||||
break;
|
||||
}
|
||||
} else if (node instanceof AST_ClassProperty) {
|
||||
if (typeof node.key == "string") add(node.key);
|
||||
} else if (node instanceof AST_Dot) {
|
||||
add(node.property);
|
||||
} else if (node instanceof AST_ObjectProperty) {
|
||||
@@ -193,6 +197,8 @@ function mangle_properties(ast, options) {
|
||||
mangleStrings(node.args[0]);
|
||||
break;
|
||||
}
|
||||
} else if (node instanceof AST_ClassProperty) {
|
||||
if (typeof node.key == "string") node.key = mangle(node.key);
|
||||
} else if (node instanceof AST_Dot) {
|
||||
node.property = mangle(node.property);
|
||||
} else if (node instanceof AST_ObjectProperty) {
|
||||
@@ -222,9 +228,7 @@ function mangle_properties(ast, options) {
|
||||
}
|
||||
|
||||
function mangle(name) {
|
||||
if (!should_mangle(name)) {
|
||||
return name;
|
||||
}
|
||||
if (!should_mangle(name)) return name;
|
||||
var mangled = cache.get(name);
|
||||
if (!mangled) {
|
||||
if (debug) {
|
||||
@@ -236,6 +240,7 @@ function mangle_properties(ast, options) {
|
||||
if (!mangled) do {
|
||||
mangled = base54(++cname);
|
||||
} while (!can_mangle(mangled));
|
||||
if (/^#/.test(name)) mangled = "#" + mangled;
|
||||
cache.set(name, mangled);
|
||||
}
|
||||
return mangled;
|
||||
|
||||
53
lib/scope.js
53
lib/scope.js
@@ -80,15 +80,16 @@ SymbolDef.prototype = {
|
||||
}
|
||||
},
|
||||
redefined: function() {
|
||||
var scope = this.defun;
|
||||
var self = this;
|
||||
var scope = self.defun;
|
||||
if (!scope) return;
|
||||
var name = this.name;
|
||||
var name = self.name;
|
||||
var def = scope.variables.get(name)
|
||||
|| scope instanceof AST_Toplevel && scope.globals.get(name)
|
||||
|| this.orig[0] instanceof AST_SymbolConst && find_if(function(def) {
|
||||
|| self.orig[0] instanceof AST_SymbolConst && find_if(function(def) {
|
||||
return def.name == name;
|
||||
}, scope.enclosed);
|
||||
if (def && def !== this) return def.redefined() || def;
|
||||
if (def && def !== self) return def.redefined() || def;
|
||||
},
|
||||
unmangleable: function(options) {
|
||||
return this.global && !options.toplevel
|
||||
@@ -96,8 +97,10 @@ SymbolDef.prototype = {
|
||||
|| this.undeclared
|
||||
|| !options.eval && this.scope.pinned()
|
||||
|| options.keep_fnames
|
||||
&& (this.orig[0] instanceof AST_SymbolLambda
|
||||
|| this.orig[0] instanceof AST_SymbolDefun);
|
||||
&& (this.orig[0] instanceof AST_SymbolClass
|
||||
|| this.orig[0] instanceof AST_SymbolDefClass
|
||||
|| this.orig[0] instanceof AST_SymbolDefun
|
||||
|| this.orig[0] instanceof AST_SymbolLambda);
|
||||
},
|
||||
};
|
||||
|
||||
@@ -115,7 +118,7 @@ function is_lhs(node, parent) {
|
||||
AST_Toplevel.DEFMETHOD("figure_out_scope", function(options) {
|
||||
options = defaults(options, {
|
||||
cache: null,
|
||||
ie8: false,
|
||||
ie: false,
|
||||
});
|
||||
|
||||
// pass 1: setup scope chaining and handle definitions
|
||||
@@ -208,7 +211,7 @@ AST_Toplevel.DEFMETHOD("figure_out_scope", function(options) {
|
||||
entangle(defun, scope);
|
||||
} else if (node instanceof AST_SymbolLambda) {
|
||||
var def = defun.def_function(node, node.name == "arguments" ? undefined : defun);
|
||||
if (options.ie8) def.defun = defun.parent_scope.resolve();
|
||||
if (options.ie) def.defun = defun.parent_scope.resolve();
|
||||
} else if (node instanceof AST_SymbolLet) {
|
||||
var def = scope.def_variable(node);
|
||||
if (exported) def.exported = true;
|
||||
@@ -271,16 +274,18 @@ AST_Toplevel.DEFMETHOD("figure_out_scope", function(options) {
|
||||
return true;
|
||||
}
|
||||
if (node instanceof AST_SymbolDeclaration) {
|
||||
var def = node.definition();
|
||||
def.preinit = def.references.length;
|
||||
if (node instanceof AST_SymbolCatch) {
|
||||
// ensure mangling works if `catch` reuses a scope variable
|
||||
var def = node.definition().redefined();
|
||||
if (def) for (var s = node.scope; s; s = s.parent_scope) {
|
||||
push_uniq(s.enclosed, def);
|
||||
if (s === def.scope) break;
|
||||
var redef = def.redefined();
|
||||
if (redef) for (var s = node.scope; s; s = s.parent_scope) {
|
||||
push_uniq(s.enclosed, redef);
|
||||
if (s === redef.scope) break;
|
||||
}
|
||||
} else if (node instanceof AST_SymbolConst) {
|
||||
// ensure compression works if `const` reuses a scope variable
|
||||
var redef = node.definition().redefined();
|
||||
var redef = def.redefined();
|
||||
if (redef) redef.const_redefs = true;
|
||||
}
|
||||
if (node.name != "arguments") return true;
|
||||
@@ -346,7 +351,7 @@ AST_Toplevel.DEFMETHOD("figure_out_scope", function(options) {
|
||||
self.walk(tw);
|
||||
|
||||
// pass 3: fix up any scoping issue with IE8
|
||||
if (options.ie8) self.walk(new TreeWalker(function(node) {
|
||||
if (options.ie) self.walk(new TreeWalker(function(node) {
|
||||
if (node instanceof AST_SymbolCatch) {
|
||||
var scope = node.thedef.defun;
|
||||
if (scope.name instanceof AST_SymbolLambda && scope.name.name == node.name) {
|
||||
@@ -392,13 +397,14 @@ AST_Toplevel.DEFMETHOD("figure_out_scope", function(options) {
|
||||
} else {
|
||||
new_def = scope.def_variable(node);
|
||||
}
|
||||
if (new_def.undeclared) self.variables.set(name, new_def);
|
||||
if (name == "arguments" && is_arguments(old_def) && node instanceof AST_SymbolLambda) return true;
|
||||
old_def.defun = new_def.scope;
|
||||
old_def.forEach(function(node) {
|
||||
node.redef = old_def;
|
||||
node.thedef = new_def;
|
||||
node.reference(options);
|
||||
});
|
||||
if (new_def.undeclared) self.variables.set(name, new_def);
|
||||
return true;
|
||||
}
|
||||
});
|
||||
@@ -566,7 +572,7 @@ AST_Symbol.DEFMETHOD("definition", function() {
|
||||
function _default_mangler_options(options) {
|
||||
options = defaults(options, {
|
||||
eval : false,
|
||||
ie8 : false,
|
||||
ie : false,
|
||||
keep_fnames : false,
|
||||
reserved : [],
|
||||
toplevel : false,
|
||||
@@ -796,6 +802,7 @@ var base54 = (function() {
|
||||
var leading = init("abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ$_");
|
||||
var chars, frequency;
|
||||
function reset() {
|
||||
chars = null;
|
||||
frequency = Object.create(freq);
|
||||
}
|
||||
base54.consider = function(str, delta) {
|
||||
@@ -807,19 +814,15 @@ var base54 = (function() {
|
||||
return frequency[b] - frequency[a];
|
||||
}
|
||||
base54.sort = function() {
|
||||
chars = leading.sort(compare).concat(digits.sort(compare));
|
||||
chars = leading.sort(compare).concat(digits).sort(compare);
|
||||
};
|
||||
base54.reset = reset;
|
||||
reset();
|
||||
function base54(num) {
|
||||
var ret = "", base = 54;
|
||||
num++;
|
||||
do {
|
||||
num--;
|
||||
ret += chars[num % base];
|
||||
num = Math.floor(num / base);
|
||||
base = 64;
|
||||
} while (num > 0);
|
||||
var ret = leading[num % 54];
|
||||
for (num = Math.floor(num / 54); --num >= 0; num >>= 6) {
|
||||
ret += chars[num & 0x3F];
|
||||
}
|
||||
return ret;
|
||||
}
|
||||
return base54;
|
||||
|
||||
@@ -3,7 +3,7 @@
|
||||
"description": "JavaScript parser, mangler/compressor and beautifier toolkit",
|
||||
"author": "Mihai Bazon <mihai.bazon@gmail.com> (http://lisperator.net/)",
|
||||
"license": "BSD-2-Clause",
|
||||
"version": "3.13.1",
|
||||
"version": "3.14.2",
|
||||
"engines": {
|
||||
"node": ">=0.8.0"
|
||||
},
|
||||
@@ -23,7 +23,7 @@
|
||||
"LICENSE"
|
||||
],
|
||||
"devDependencies": {
|
||||
"acorn": "~7.1.0",
|
||||
"acorn": "~8.2.1",
|
||||
"semver": "~6.3.0"
|
||||
},
|
||||
"scripts": {
|
||||
|
||||
@@ -16,7 +16,7 @@ var urls = [
|
||||
"https://code.angularjs.org/1.7.8/angular.js",
|
||||
"https://unpkg.com/mathjs@6.2.3/dist/math.js",
|
||||
"https://unpkg.com/react@15.3.2/dist/react.js",
|
||||
"https://cdnjs.cloudflare.com/ajax/libs/d3/5.12.0/d3.js",
|
||||
"https://cdnjs.cloudflare.com/ajax/libs/d3/6.7.0/d3.js",
|
||||
"https://maxcdn.bootstrapcdn.com/bootstrap/3.3.7/js/bootstrap.js",
|
||||
"https://cdnjs.cloudflare.com/ajax/libs/lodash.js/4.17.15/lodash.js",
|
||||
"https://cdnjs.cloudflare.com/ajax/libs/ember.js/2.12.2/ember.prod.js",
|
||||
|
||||
@@ -262,6 +262,7 @@ function test_case(test) {
|
||||
var input = to_toplevel(test.input, test.mangle);
|
||||
var input_code = make_code(input);
|
||||
var input_formatted = make_code(test.input, {
|
||||
annotations: true,
|
||||
beautify: true,
|
||||
comments: "all",
|
||||
keep_quoted_props: true,
|
||||
|
||||
474
test/compress/annotations.js
Normal file
474
test/compress/annotations.js
Normal file
@@ -0,0 +1,474 @@
|
||||
issue_2629_1: {
|
||||
options = {
|
||||
annotations: true,
|
||||
side_effects: true,
|
||||
}
|
||||
beautify = {
|
||||
comments: "all",
|
||||
}
|
||||
input: {
|
||||
/*@__PURE__*/ a();
|
||||
/*@__PURE__*/ (b());
|
||||
(/*@__PURE__*/ c)();
|
||||
(/*@__PURE__*/ d());
|
||||
}
|
||||
expect_exact: "c();"
|
||||
}
|
||||
|
||||
issue_2629_2: {
|
||||
options = {
|
||||
annotations: true,
|
||||
side_effects: true,
|
||||
}
|
||||
beautify = {
|
||||
comments: "all",
|
||||
}
|
||||
input: {
|
||||
/*@__PURE__*/ a(1)(2)(3);
|
||||
/*@__PURE__*/ (b(1))(2)(3);
|
||||
/*@__PURE__*/ (c(1)(2))(3);
|
||||
/*@__PURE__*/ (d(1)(2)(3));
|
||||
(/*@__PURE__*/ e)(1)(2)(3);
|
||||
(/*@__PURE__*/ f(1))(2)(3);
|
||||
(/*@__PURE__*/ g(1)(2))(3);
|
||||
(/*@__PURE__*/ h(1)(2)(3));
|
||||
}
|
||||
expect_exact: [
|
||||
"e(1)(2)(3);",
|
||||
"f(1)(2)(3);",
|
||||
"g(1)(2)(3);",
|
||||
]
|
||||
}
|
||||
|
||||
issue_2629_3: {
|
||||
options = {
|
||||
annotations: true,
|
||||
side_effects: true,
|
||||
}
|
||||
beautify = {
|
||||
comments: "all",
|
||||
}
|
||||
input: {
|
||||
/*@__PURE__*/ a.x(1).y(2).z(3);
|
||||
/*@__PURE__*/ (b.x)(1).y(2).z(3);
|
||||
/*@__PURE__*/ (c.x(1)).y(2).z(3);
|
||||
/*@__PURE__*/ (d.x(1).y)(2).z(3);
|
||||
/*@__PURE__*/ (e.x(1).y(2)).z(3);
|
||||
/*@__PURE__*/ (f.x(1).y(2).z)(3);
|
||||
/*@__PURE__*/ (g.x(1).y(2).z(3));
|
||||
(/*@__PURE__*/ h).x(1).y(2).z(3);
|
||||
(/*@__PURE__*/ i.x)(1).y(2).z(3);
|
||||
(/*@__PURE__*/ j.x(1)).y(2).z(3);
|
||||
(/*@__PURE__*/ k.x(1).y)(2).z(3);
|
||||
(/*@__PURE__*/ l.x(1).y(2)).z(3);
|
||||
(/*@__PURE__*/ m.x(1).y(2).z)(3);
|
||||
(/*@__PURE__*/ n.x(1).y(2).z(3));
|
||||
}
|
||||
expect_exact: [
|
||||
"h.x(1).y(2).z(3);",
|
||||
"i.x(1).y(2).z(3);",
|
||||
"j.x(1).y(2).z(3);",
|
||||
"k.x(1).y(2).z(3);",
|
||||
"l.x(1).y(2).z(3);",
|
||||
"m.x(1).y(2).z(3);",
|
||||
]
|
||||
}
|
||||
|
||||
issue_2629_4: {
|
||||
options = {
|
||||
annotations: true,
|
||||
side_effects: true,
|
||||
}
|
||||
input: {
|
||||
(/*@__PURE__*/ x(), y());
|
||||
(w(), /*@__PURE__*/ x(), y());
|
||||
}
|
||||
expect: {
|
||||
y();
|
||||
w(), y();
|
||||
}
|
||||
}
|
||||
|
||||
issue_2629_5: {
|
||||
options = {
|
||||
annotations: true,
|
||||
side_effects: true,
|
||||
}
|
||||
input: {
|
||||
[ /*@__PURE__*/ x() ];
|
||||
[ /*@__PURE__*/ x(), y() ];
|
||||
[ w(), /*@__PURE__*/ x(), y() ];
|
||||
}
|
||||
expect: {
|
||||
y();
|
||||
w(), y();
|
||||
}
|
||||
}
|
||||
|
||||
issue_2638: {
|
||||
options = {
|
||||
annotations: true,
|
||||
side_effects: true,
|
||||
}
|
||||
beautify = {
|
||||
comments: "all",
|
||||
}
|
||||
input: {
|
||||
/*@__PURE__*/(g() || h())(x(), y());
|
||||
(/*@__PURE__*/ (a() || b()))(c(), d());
|
||||
}
|
||||
expect_exact: [
|
||||
"x(),y();",
|
||||
"(a()||b())(c(),d());",
|
||||
]
|
||||
}
|
||||
|
||||
issue_2705_1: {
|
||||
options = {
|
||||
annotations: true,
|
||||
side_effects: true,
|
||||
}
|
||||
beautify = {
|
||||
comments: "all",
|
||||
}
|
||||
input: {
|
||||
/*@__PURE__*/ new a();
|
||||
/*@__PURE__*/ (new b());
|
||||
new (/*@__PURE__*/ c)();
|
||||
(/*@__PURE__*/ new d());
|
||||
}
|
||||
expect_exact: [
|
||||
"new c;",
|
||||
]
|
||||
}
|
||||
|
||||
issue_2705_2: {
|
||||
options = {
|
||||
annotations: true,
|
||||
side_effects: true,
|
||||
}
|
||||
beautify = {
|
||||
comments: "all",
|
||||
}
|
||||
input: {
|
||||
/*@__PURE__*/ new a(1)(2)(3);
|
||||
/*@__PURE__*/ new (b(1))(2)(3);
|
||||
/*@__PURE__*/ new (c(1)(2))(3);
|
||||
/*@__PURE__*/ new (d(1)(2)(3));
|
||||
new (/*@__PURE__*/ e)(1)(2)(3);
|
||||
(/*@__PURE__*/ new f(1))(2)(3);
|
||||
(/*@__PURE__*/ new g(1)(2))(3);
|
||||
(/*@__PURE__*/ new h(1)(2)(3));
|
||||
}
|
||||
expect_exact: [
|
||||
"new e(1)(2)(3);",
|
||||
"new f(1)(2)(3);",
|
||||
"new g(1)(2)(3);",
|
||||
]
|
||||
}
|
||||
|
||||
issue_2705_3: {
|
||||
options = {
|
||||
annotations: true,
|
||||
side_effects: true,
|
||||
}
|
||||
beautify = {
|
||||
comments: "all",
|
||||
}
|
||||
input: {
|
||||
/*@__PURE__*/ new a.x(1).y(2).z(3);
|
||||
/*@__PURE__*/ new (b.x)(1).y(2).z(3);
|
||||
/*@__PURE__*/ new (c.x(1)).y(2).z(3);
|
||||
/*@__PURE__*/ new (d.x(1).y)(2).z(3);
|
||||
/*@__PURE__*/ new (e.x(1).y(2)).z(3);
|
||||
/*@__PURE__*/ new (f.x(1).y(2).z)(3);
|
||||
/*@__PURE__*/ new (g.x(1).y(2).z(3));
|
||||
new (/*@__PURE__*/ h).x(1).y(2).z(3);
|
||||
/* */ new (/*@__PURE__*/ i.x)(1).y(2).z(3);
|
||||
(/*@__PURE__*/ new j.x(1)).y(2).z(3);
|
||||
(/*@__PURE__*/ new k.x(1).y)(2).z(3);
|
||||
(/*@__PURE__*/ new l.x(1).y(2)).z(3);
|
||||
(/*@__PURE__*/ new m.x(1).y(2).z)(3);
|
||||
(/*@__PURE__*/ new n.x(1).y(2).z(3));
|
||||
}
|
||||
expect_exact: [
|
||||
"new h.x(1).y(2).z(3);",
|
||||
"/* */new i.x(1).y(2).z(3);",
|
||||
"new j.x(1).y(2).z(3);",
|
||||
"new k.x(1).y(2).z(3);",
|
||||
"new l.x(1).y(2).z(3);",
|
||||
"new m.x(1).y(2).z(3);",
|
||||
]
|
||||
}
|
||||
|
||||
issue_2705_4: {
|
||||
options = {
|
||||
annotations: true,
|
||||
side_effects: true,
|
||||
}
|
||||
input: {
|
||||
(/*@__PURE__*/ new x(), y());
|
||||
(w(), /*@__PURE__*/ new x(), y());
|
||||
}
|
||||
expect: {
|
||||
y();
|
||||
w(), y();
|
||||
}
|
||||
}
|
||||
|
||||
issue_2705_5: {
|
||||
options = {
|
||||
annotations: true,
|
||||
side_effects: true,
|
||||
}
|
||||
input: {
|
||||
[ /*@__PURE__*/ new x() ];
|
||||
[ /*@__PURE__*/ new x(), y() ];
|
||||
[ w(), /*@__PURE__*/ new x(), y() ];
|
||||
}
|
||||
expect: {
|
||||
y();
|
||||
w(), y();
|
||||
}
|
||||
}
|
||||
|
||||
issue_2705_6: {
|
||||
options = {
|
||||
annotations: true,
|
||||
side_effects: true,
|
||||
}
|
||||
beautify = {
|
||||
comments: "all",
|
||||
}
|
||||
input: {
|
||||
/*@__PURE__*/new (g() || h())(x(), y());
|
||||
/* */ new (/*@__PURE__*/ (a() || b()))(c(), d());
|
||||
}
|
||||
expect_exact: [
|
||||
"x(),y();",
|
||||
"/* */new(a()||b())(c(),d());",
|
||||
]
|
||||
}
|
||||
|
||||
issue_3858: {
|
||||
options = {
|
||||
annotations: true,
|
||||
collapse_vars: true,
|
||||
inline: true,
|
||||
keep_fargs: false,
|
||||
unsafe: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var f = function(a) {
|
||||
return /*@__PURE__*/ function(b) {
|
||||
console.log(b);
|
||||
}(a);
|
||||
};
|
||||
f("PASS");
|
||||
}
|
||||
expect: {
|
||||
var f = function(a) {
|
||||
return function() {
|
||||
console.log(a);
|
||||
}();
|
||||
};
|
||||
f("PASS");
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
inline_pure_call_1: {
|
||||
options = {
|
||||
annotations: true,
|
||||
collapse_vars: true,
|
||||
inline: true,
|
||||
keep_fargs: false,
|
||||
reduce_vars: true,
|
||||
sequences: true,
|
||||
side_effects: true,
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var f = function(a) {
|
||||
return /*@__PURE__*/ function(b) {
|
||||
console.log(b);
|
||||
}(a);
|
||||
};
|
||||
f("PASS");
|
||||
}
|
||||
expect: {}
|
||||
}
|
||||
|
||||
inline_pure_call_2: {
|
||||
options = {
|
||||
annotations: true,
|
||||
collapse_vars: true,
|
||||
inline: true,
|
||||
keep_fargs: false,
|
||||
reduce_vars: true,
|
||||
sequences: true,
|
||||
side_effects: true,
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var f = function(a) {
|
||||
return /*@__PURE__*/ function(b) {
|
||||
console.log(b);
|
||||
}(a);
|
||||
};
|
||||
var a = f("PASS");
|
||||
}
|
||||
expect: {}
|
||||
}
|
||||
|
||||
inline_pure_call_3: {
|
||||
options = {
|
||||
annotations: true,
|
||||
collapse_vars: true,
|
||||
evaluate: true,
|
||||
inline: true,
|
||||
keep_fargs: false,
|
||||
passes: 2,
|
||||
reduce_vars: true,
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var f = function(a) {
|
||||
return /*@__PURE__*/ function(b) {
|
||||
console.log(b);
|
||||
}(a);
|
||||
};
|
||||
var a = f("PASS");
|
||||
console.log(a);
|
||||
}
|
||||
expect: {
|
||||
var a = function() {
|
||||
console.log("PASS");
|
||||
}();
|
||||
console.log(a);
|
||||
}
|
||||
expect_stdout: [
|
||||
"PASS",
|
||||
"undefined",
|
||||
]
|
||||
}
|
||||
|
||||
inline_pure_call_4: {
|
||||
options = {
|
||||
annotations: true,
|
||||
evaluate: true,
|
||||
reduce_vars: true,
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var a = /*@__PURE__*/ function() {
|
||||
return console.log("PASS"), 42;
|
||||
}();
|
||||
console.log(a);
|
||||
}
|
||||
expect: {
|
||||
var a = function() {
|
||||
return console.log("PASS"), 42;
|
||||
}();
|
||||
console.log(a);
|
||||
}
|
||||
expect_stdout: [
|
||||
"PASS",
|
||||
"42",
|
||||
]
|
||||
}
|
||||
|
||||
compress_and_output_annotations_enabled: {
|
||||
options = {
|
||||
annotations: true,
|
||||
side_effects: true,
|
||||
}
|
||||
beautify = {
|
||||
annotations: true,
|
||||
beautify: true,
|
||||
comments: false,
|
||||
}
|
||||
input: {
|
||||
var top = /*@__PURE__*/ foo();
|
||||
/*@__PURE__*/ a(1)(2)(3);
|
||||
/*@__PURE__*/ (b(1))(2)(3);
|
||||
/*@__PURE__*/ (c(1)(2))(3);
|
||||
/*@__PURE__*/ (d(1)(2)(3));
|
||||
(/*@__PURE__*/ e)(1)(2)(3);
|
||||
(/*@__PURE__*/ f(1))(2)(3);
|
||||
(/*@__PURE__*/ g(1)(2))(3);
|
||||
(/*@__PURE__*/ h(1)(2)(3));
|
||||
/*@__PURE__*/ l(1).p(2);
|
||||
(/*@__PURE__*/ m(1)).p(2);
|
||||
(/*@__PURE__*/ n(1).p)(2);
|
||||
(/*@__PURE__*/ o(1).p(2));
|
||||
}
|
||||
expect_exact: [
|
||||
"var top = /*@__PURE__*/foo();",
|
||||
"",
|
||||
"e(1)(2)(3);",
|
||||
"",
|
||||
"f(1)(2)(3);",
|
||||
"",
|
||||
"g(1)(2)(3);",
|
||||
"",
|
||||
"m(1).p(2);",
|
||||
"",
|
||||
"n(1).p(2);",
|
||||
]
|
||||
}
|
||||
|
||||
compress_annotations_disabled_output_annotations_enabled: {
|
||||
options = {
|
||||
annotations: false,
|
||||
evaluate: true,
|
||||
sequences: true,
|
||||
side_effects: true,
|
||||
}
|
||||
beautify = {
|
||||
annotations: true,
|
||||
comments: true,
|
||||
}
|
||||
input: {
|
||||
/*@__PURE__*/ a(1+2);
|
||||
/*#__PURE__*/ (b(2+3));
|
||||
(/*@__PURE__*/ c)(side_effect);
|
||||
(/*#__PURE__*/ d(effect()));
|
||||
}
|
||||
expect_exact: [
|
||||
"/*@__PURE__*/a(3),",
|
||||
"/*#__PURE__*/b(5),",
|
||||
"c(side_effect),",
|
||||
"/*#__PURE__*/d(effect());",
|
||||
]
|
||||
}
|
||||
|
||||
compress_and_output_annotations_disabled: {
|
||||
options = {
|
||||
annotations: false,
|
||||
evaluate: true,
|
||||
sequences: true,
|
||||
side_effects: true,
|
||||
}
|
||||
beautify = {
|
||||
annotations: false,
|
||||
comments: true,
|
||||
}
|
||||
input: {
|
||||
/*@__PURE__*/ a(1+2);
|
||||
/*@__PURE__*/ (b(2+3));
|
||||
(/*@__PURE__*/ c)(side_effect);
|
||||
(/*@__PURE__*/ d(effect()));
|
||||
}
|
||||
expect_exact: [
|
||||
"a(3),",
|
||||
"b(5),",
|
||||
"c(side_effect),",
|
||||
"d(effect());",
|
||||
]
|
||||
}
|
||||
@@ -84,6 +84,7 @@ replace_index_drop_fargs_1: {
|
||||
evaluate: true,
|
||||
keep_fargs: false,
|
||||
properties: true,
|
||||
reduce_vars: true,
|
||||
}
|
||||
input: {
|
||||
var arguments = [];
|
||||
@@ -119,7 +120,7 @@ replace_index_drop_fargs_1: {
|
||||
console.log(b, b, arguments.foo);
|
||||
})("bar", 42);
|
||||
(function(arguments) {
|
||||
console.log(arguments[1], arguments[1], arguments.foo);
|
||||
console.log("bar"[1], "bar"[1], "bar".foo);
|
||||
})("bar", 42);
|
||||
(function(argument_0, argument_1) {
|
||||
var arguments;
|
||||
@@ -253,6 +254,25 @@ duplicate_argname: {
|
||||
expect_stdout: "bar 42 foo 42 bar"
|
||||
}
|
||||
|
||||
fraction: {
|
||||
options = {
|
||||
arguments: true,
|
||||
keep_fargs: false,
|
||||
reduce_vars: true,
|
||||
}
|
||||
input: {
|
||||
console.log(function() {
|
||||
return arguments[0.3];
|
||||
}("FAIL") || "PASS");
|
||||
}
|
||||
expect: {
|
||||
console.log(function() {
|
||||
return arguments[0.3];
|
||||
}("FAIL") || "PASS");
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
issue_3273: {
|
||||
options = {
|
||||
arguments: true,
|
||||
@@ -649,6 +669,7 @@ issue_3420_1: {
|
||||
options = {
|
||||
arguments: true,
|
||||
keep_fargs: false,
|
||||
reduce_vars: true,
|
||||
}
|
||||
input: {
|
||||
console.log(function() {
|
||||
@@ -671,6 +692,7 @@ issue_3420_2: {
|
||||
options = {
|
||||
arguments: true,
|
||||
keep_fargs: false,
|
||||
reduce_vars: true,
|
||||
}
|
||||
input: {
|
||||
var foo = function() {
|
||||
@@ -691,6 +713,7 @@ issue_3420_3: {
|
||||
options = {
|
||||
arguments: true,
|
||||
keep_fargs: false,
|
||||
reduce_vars: true,
|
||||
}
|
||||
input: {
|
||||
"use strict";
|
||||
@@ -713,6 +736,7 @@ issue_3420_4: {
|
||||
options = {
|
||||
arguments: true,
|
||||
keep_fargs: false,
|
||||
reduce_vars: true,
|
||||
}
|
||||
input: {
|
||||
!function() {
|
||||
@@ -738,6 +762,7 @@ issue_3420_5: {
|
||||
options = {
|
||||
arguments: true,
|
||||
keep_fargs: false,
|
||||
reduce_vars: true,
|
||||
}
|
||||
input: {
|
||||
"use strict";
|
||||
@@ -765,6 +790,7 @@ issue_3420_6: {
|
||||
options = {
|
||||
arguments: true,
|
||||
keep_fargs: false,
|
||||
reduce_vars: true,
|
||||
}
|
||||
input: {
|
||||
console.log(function() {
|
||||
@@ -783,6 +809,7 @@ issue_3420_7: {
|
||||
options = {
|
||||
arguments: true,
|
||||
keep_fargs: false,
|
||||
reduce_vars: true,
|
||||
}
|
||||
input: {
|
||||
"use strict";
|
||||
@@ -827,6 +854,7 @@ issue_4291_1: {
|
||||
options = {
|
||||
arguments: true,
|
||||
keep_fargs: false,
|
||||
reduce_vars: true,
|
||||
}
|
||||
input: {
|
||||
console.log(function() {
|
||||
@@ -847,6 +875,7 @@ issue_4291_2: {
|
||||
options = {
|
||||
arguments: true,
|
||||
keep_fargs: false,
|
||||
reduce_vars: true,
|
||||
}
|
||||
input: {
|
||||
var a = function() {
|
||||
@@ -857,8 +886,8 @@ issue_4291_2: {
|
||||
console.log(a[1], a[0], a.length);
|
||||
}
|
||||
expect: {
|
||||
var a = function(argument_0) {
|
||||
if (argument_0)
|
||||
var a = function() {
|
||||
if (arguments[0])
|
||||
arguments[1] = "PASS";
|
||||
return arguments;
|
||||
}(42);
|
||||
@@ -871,6 +900,7 @@ issue_4397: {
|
||||
options = {
|
||||
arguments: true,
|
||||
keep_fargs: false,
|
||||
reduce_vars: true,
|
||||
}
|
||||
input: {
|
||||
console.log(typeof function() {
|
||||
@@ -999,3 +1029,26 @@ issue_4696: {
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
issue_4809: {
|
||||
options = {
|
||||
arguments: true,
|
||||
keep_fargs: false,
|
||||
reduce_vars: true,
|
||||
}
|
||||
input: {
|
||||
A = 0;
|
||||
(function() {
|
||||
arguments[A] = "PASS";
|
||||
console.log(arguments[0]);
|
||||
})();
|
||||
}
|
||||
expect: {
|
||||
A = 0;
|
||||
(function() {
|
||||
arguments[A] = "PASS";
|
||||
console.log(arguments[0]);
|
||||
})();
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
@@ -276,6 +276,7 @@ drop_arguments: {
|
||||
options = {
|
||||
arguments: true,
|
||||
keep_fargs: false,
|
||||
reduce_vars: true,
|
||||
}
|
||||
input: {
|
||||
console.log(function() {
|
||||
@@ -420,6 +421,7 @@ collapse_value: {
|
||||
arrows: true,
|
||||
collapse_vars: true,
|
||||
keep_fargs: false,
|
||||
unsafe: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
@@ -554,6 +556,75 @@ reduce_iife_3: {
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
reduce_lambda_1: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
reduce_vars: true,
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var f = () => {
|
||||
console.log(a, b);
|
||||
};
|
||||
var a = "foo", b = 42;
|
||||
f();
|
||||
b = "bar";
|
||||
f();
|
||||
}
|
||||
expect: {
|
||||
var f = () => {
|
||||
console.log("foo", b);
|
||||
};
|
||||
var b = 42;
|
||||
f();
|
||||
b = "bar";
|
||||
f();
|
||||
}
|
||||
expect_stdout: [
|
||||
"foo 42",
|
||||
"foo bar",
|
||||
]
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
reduce_lambda_2: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
passes: 2,
|
||||
reduce_vars: true,
|
||||
side_effects: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
(function(f, a, b) {
|
||||
f = () => {
|
||||
console.log(a, b);
|
||||
};
|
||||
a = "foo", b = 42;
|
||||
f();
|
||||
b = "bar";
|
||||
f();
|
||||
})();
|
||||
}
|
||||
expect: {
|
||||
(function(f, a, b) {
|
||||
f = () => {
|
||||
console.log("foo", b);
|
||||
};
|
||||
b = 42;
|
||||
f();
|
||||
b = "bar";
|
||||
f();
|
||||
})();
|
||||
}
|
||||
expect_stdout: [
|
||||
"foo 42",
|
||||
"foo bar",
|
||||
]
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
single_use_recursive: {
|
||||
options = {
|
||||
reduce_vars: true,
|
||||
@@ -803,3 +874,13 @@ issue_4687_2: {
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
issue_4772: {
|
||||
input: {
|
||||
var f = a => (a)
|
||||
/**/ console.log(f("PASS"));
|
||||
}
|
||||
expect_exact: 'var f=a=>a;console.log(f("PASS"));'
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
@@ -475,3 +475,274 @@ issue_4521: {
|
||||
}
|
||||
expect_stdout: "42"
|
||||
}
|
||||
|
||||
logical_assignments: {
|
||||
input: {
|
||||
var a = 42, b = null, c;
|
||||
a &&= "foo";
|
||||
b ||= "bar";
|
||||
c ??= "baz";
|
||||
console.log(a, b, c);
|
||||
}
|
||||
expect_exact: 'var a=42,b=null,c;a&&="foo";b||="bar";c??="baz";console.log(a,b,c);'
|
||||
expect_stdout: "foo bar baz"
|
||||
node_version: ">=15"
|
||||
}
|
||||
|
||||
logical_collapse_vars: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
}
|
||||
input: {
|
||||
var a = "FAIL", b = false;
|
||||
a = "PASS";
|
||||
b ??= a;
|
||||
console.log(a);
|
||||
}
|
||||
expect: {
|
||||
var a = "FAIL", b = false;
|
||||
a = "PASS";
|
||||
b ??= a;
|
||||
console.log(a);
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=15"
|
||||
}
|
||||
|
||||
logical_reduce_vars: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
reduce_vars: true,
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var a = "PASS", b = 42;
|
||||
b ??= a = "FAIL";
|
||||
console.log(a);
|
||||
}
|
||||
expect: {
|
||||
var a = "PASS", b = 42;
|
||||
b ??= a = "FAIL";
|
||||
console.log(a);
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=15"
|
||||
}
|
||||
|
||||
logical_side_effects: {
|
||||
options = {
|
||||
side_effects: true,
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var a = "PASS", b = 42;
|
||||
b ??= a = "FAIL";
|
||||
console.log(a);
|
||||
}
|
||||
expect: {
|
||||
var a = "PASS", b = 42;
|
||||
b ??= a = "FAIL";
|
||||
console.log(a);
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=15"
|
||||
}
|
||||
|
||||
issue_4815_1: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
reduce_vars: true,
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var a = "PASS";
|
||||
42..p &&= a = "FAIL";
|
||||
console.log(a);
|
||||
}
|
||||
expect: {
|
||||
var a = "PASS";
|
||||
42..p &&= a = "FAIL";
|
||||
console.log(a);
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=15"
|
||||
}
|
||||
|
||||
issue_4815_2: {
|
||||
options = {
|
||||
pure_getters: "strict",
|
||||
side_effects: true,
|
||||
}
|
||||
input: {
|
||||
var a = "PASS";
|
||||
42..p &&= a = "FAIL";
|
||||
console.log(a);
|
||||
}
|
||||
expect: {
|
||||
var a = "PASS";
|
||||
42..p &&= a = "FAIL";
|
||||
console.log(a);
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=15"
|
||||
}
|
||||
|
||||
issue_4819: {
|
||||
options = {
|
||||
comparisons: true,
|
||||
}
|
||||
input: {
|
||||
console.log(void 0 === ([].p &&= 42));
|
||||
}
|
||||
expect: {
|
||||
console.log(void 0 === ([].p &&= 42));
|
||||
}
|
||||
expect_stdout: "true"
|
||||
node_version: ">=15"
|
||||
}
|
||||
|
||||
issue_4827_1: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
toplevel: true,
|
||||
}
|
||||
input: {
|
||||
A = "FAIL";
|
||||
var a = A, b = "PASS", c;
|
||||
c &&= b = a, console.log(b);
|
||||
}
|
||||
expect: {
|
||||
A = "FAIL";
|
||||
var a = A, b = "PASS", c;
|
||||
c &&= b = a, console.log(b);
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=15"
|
||||
}
|
||||
|
||||
issue_4827_2: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
inline: true,
|
||||
reduce_vars: true,
|
||||
side_effects: true,
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var a = 0, b = "PASS";
|
||||
function f(c) {
|
||||
a++,
|
||||
c &&= b = a;
|
||||
}
|
||||
f();
|
||||
console.log(b);
|
||||
}
|
||||
expect: {
|
||||
var a = 0, b = "PASS";
|
||||
a++,
|
||||
c &&= b = a;
|
||||
var c;
|
||||
console.log(b);
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=15"
|
||||
}
|
||||
|
||||
issue_4827_3: {
|
||||
options = {
|
||||
merge_vars: true,
|
||||
toplevel: true,
|
||||
}
|
||||
input: {
|
||||
var a = 0, b, c;
|
||||
a++;
|
||||
c &&= b = a;
|
||||
console.log(b);
|
||||
}
|
||||
expect: {
|
||||
var a = 0, b, c;
|
||||
a++;
|
||||
c &&= b = a;
|
||||
console.log(b);
|
||||
}
|
||||
expect_stdout: "undefined"
|
||||
node_version: ">=15"
|
||||
}
|
||||
|
||||
issue_4876: {
|
||||
options = {
|
||||
pure_getters: "strict",
|
||||
reduce_vars: true,
|
||||
side_effects: true,
|
||||
toplevel: true,
|
||||
}
|
||||
input: {
|
||||
try {
|
||||
var a = null;
|
||||
var b = a &&= 42;
|
||||
b.p;
|
||||
} catch (e) {
|
||||
console.log("PASS");
|
||||
}
|
||||
}
|
||||
expect: {
|
||||
try {
|
||||
var a = null;
|
||||
var b = a &&= 42;
|
||||
b.p;
|
||||
} catch (e) {
|
||||
console.log("PASS");
|
||||
}
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=15"
|
||||
}
|
||||
|
||||
issue_4924_1: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
side_effects: true,
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var a, b;
|
||||
console.log("PASS");
|
||||
a = function() {};
|
||||
b = function() {}(b ||= a);
|
||||
}
|
||||
expect: {
|
||||
var b;
|
||||
console.log("PASS");
|
||||
b = void (b ||= function() {});
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=15"
|
||||
}
|
||||
|
||||
issue_4924_2: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
dead_code: true,
|
||||
passes: 2,
|
||||
sequences: true,
|
||||
side_effects: true,
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var a, b;
|
||||
console.log("PASS");
|
||||
a = function() {};
|
||||
b = function() {}(b ||= a);
|
||||
}
|
||||
expect: {
|
||||
console.log("PASS");
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=15"
|
||||
}
|
||||
|
||||
@@ -227,9 +227,7 @@ inline_await_1_trim: {
|
||||
console.log("PASS");
|
||||
}
|
||||
expect: {
|
||||
(async function() {
|
||||
await 0;
|
||||
})();
|
||||
0;
|
||||
console.log("PASS");
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
@@ -334,7 +332,7 @@ inline_await_3_trim: {
|
||||
}
|
||||
expect: {
|
||||
(async function() {
|
||||
return a = "PASS", b = console.log, await b(a);
|
||||
return a = "PASS", b = console.log, b(a);
|
||||
var a, b;
|
||||
})();
|
||||
}
|
||||
@@ -557,6 +555,51 @@ collapse_property_lambda: {
|
||||
node_version: ">=8"
|
||||
}
|
||||
|
||||
drop_async_1: {
|
||||
options = {
|
||||
awaits: true,
|
||||
inline: true,
|
||||
side_effects: true,
|
||||
}
|
||||
input: {
|
||||
console.log(function(a) {
|
||||
(async function() {
|
||||
a *= 7;
|
||||
})();
|
||||
return a;
|
||||
}(6));
|
||||
}
|
||||
expect: {
|
||||
console.log(function(a) {
|
||||
a *= 7;
|
||||
return a;
|
||||
}(6));
|
||||
}
|
||||
expect_stdout: "42"
|
||||
node_version: ">=8"
|
||||
}
|
||||
|
||||
drop_async_2: {
|
||||
options = {
|
||||
awaits: true,
|
||||
collapse_vars: true,
|
||||
evaluate: true,
|
||||
inline: true,
|
||||
side_effects: true,
|
||||
}
|
||||
input: {
|
||||
console.log(function(a) {
|
||||
(async b => await (a *= b))(7);
|
||||
return a;
|
||||
}(6));
|
||||
}
|
||||
expect: {
|
||||
console.log(42);
|
||||
}
|
||||
expect_stdout: "42"
|
||||
node_version: ">=8"
|
||||
}
|
||||
|
||||
drop_return: {
|
||||
options = {
|
||||
side_effects: true,
|
||||
@@ -564,7 +607,7 @@ drop_return: {
|
||||
input: {
|
||||
(async function(a) {
|
||||
while (!console);
|
||||
return console.log(a);
|
||||
return !console.log(a);
|
||||
})(42);
|
||||
}
|
||||
expect: {
|
||||
@@ -614,14 +657,14 @@ functions: {
|
||||
async function b() {
|
||||
return !!b;
|
||||
}
|
||||
var c = async function(c) {
|
||||
async function c(c) {
|
||||
return c;
|
||||
};
|
||||
}
|
||||
if (await c(await b(await a()))) {
|
||||
async function d() {}
|
||||
async function e() {
|
||||
return typeof e;
|
||||
}
|
||||
var d = async function() {};
|
||||
var e = async function y() {
|
||||
return typeof y;
|
||||
};
|
||||
var f = async function(f) {
|
||||
return f;
|
||||
};
|
||||
@@ -672,9 +715,9 @@ functions_use_strict: {
|
||||
async function b() {
|
||||
return !!b;
|
||||
}
|
||||
var c = async function(c) {
|
||||
async function c(c) {
|
||||
return c;
|
||||
};
|
||||
}
|
||||
if (await c(await b(await a()))) {
|
||||
var d = async function() {};
|
||||
var e = async function y() {
|
||||
@@ -691,6 +734,54 @@ functions_use_strict: {
|
||||
node_version: ">=8"
|
||||
}
|
||||
|
||||
functions_anonymous: {
|
||||
options = {
|
||||
functions: true,
|
||||
reduce_vars: true,
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var await = async function() {
|
||||
console.log("PASS");
|
||||
};
|
||||
await(await);
|
||||
}
|
||||
expect: {
|
||||
async function await() {
|
||||
console.log("PASS");
|
||||
}
|
||||
await();
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=8"
|
||||
}
|
||||
|
||||
functions_inner_var: {
|
||||
options = {
|
||||
functions: true,
|
||||
reduce_vars: true,
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var await = function a() {
|
||||
var a;
|
||||
console.log(a, a);
|
||||
};
|
||||
await(await);
|
||||
}
|
||||
expect: {
|
||||
function await() {
|
||||
var a;
|
||||
console.log(a, a);
|
||||
}
|
||||
await();
|
||||
}
|
||||
expect_stdout: "undefined undefined"
|
||||
node_version: ">=8"
|
||||
}
|
||||
|
||||
issue_4335_1: {
|
||||
options = {
|
||||
inline: true,
|
||||
@@ -1408,3 +1499,550 @@ issue_4747: {
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=8"
|
||||
}
|
||||
|
||||
issue_4764_1: {
|
||||
options = {
|
||||
side_effects: true,
|
||||
}
|
||||
input: {
|
||||
(async function() {
|
||||
return {
|
||||
then() {
|
||||
console.log("PASS");
|
||||
},
|
||||
};
|
||||
})();
|
||||
}
|
||||
expect: {
|
||||
(async function() {
|
||||
return {
|
||||
then() {
|
||||
console.log("PASS");
|
||||
},
|
||||
};
|
||||
})();
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=8"
|
||||
}
|
||||
|
||||
issue_4764_2: {
|
||||
options = {
|
||||
arrows: true,
|
||||
side_effects: true,
|
||||
}
|
||||
input: {
|
||||
(async () => ({
|
||||
get then() {
|
||||
console.log("PASS");
|
||||
},
|
||||
}))();
|
||||
}
|
||||
expect: {
|
||||
(async () => ({
|
||||
get then() {
|
||||
console.log("PASS");
|
||||
},
|
||||
}))();
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=8"
|
||||
}
|
||||
|
||||
issue_4764_3: {
|
||||
options = {
|
||||
side_effects: true,
|
||||
}
|
||||
input: {
|
||||
(async function(o) {
|
||||
return o;
|
||||
})({
|
||||
then() {
|
||||
console.log("PASS");
|
||||
},
|
||||
});
|
||||
}
|
||||
expect: {
|
||||
(async function(o) {
|
||||
return o;
|
||||
})({
|
||||
then() {
|
||||
console.log("PASS");
|
||||
},
|
||||
});
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=8"
|
||||
}
|
||||
|
||||
issue_4972_1: {
|
||||
options = {
|
||||
awaits: true,
|
||||
side_effects: true,
|
||||
}
|
||||
input: {
|
||||
console.log("foo");
|
||||
(async function() {
|
||||
try {
|
||||
return await "bar";
|
||||
} finally {
|
||||
console.log("baz");
|
||||
}
|
||||
})().then(console.log);
|
||||
console.log("moo");
|
||||
}
|
||||
expect: {
|
||||
console.log("foo");
|
||||
(async function() {
|
||||
try {
|
||||
return await "bar";
|
||||
} finally {
|
||||
console.log("baz");
|
||||
}
|
||||
})().then(console.log);
|
||||
console.log("moo");
|
||||
}
|
||||
expect_stdout: [
|
||||
"foo",
|
||||
"moo",
|
||||
"baz",
|
||||
"bar",
|
||||
]
|
||||
node_version: ">=8"
|
||||
}
|
||||
|
||||
issue_4972_2: {
|
||||
options = {
|
||||
awaits: true,
|
||||
side_effects: true,
|
||||
}
|
||||
input: {
|
||||
console.log("foo");
|
||||
(async function() {
|
||||
try {
|
||||
console.log("bar");
|
||||
} finally {
|
||||
return await "baz";
|
||||
}
|
||||
})().then(console.log);
|
||||
console.log("moo");
|
||||
}
|
||||
expect: {
|
||||
console.log("foo");
|
||||
(async function() {
|
||||
try {
|
||||
console.log("bar");
|
||||
} finally {
|
||||
return "baz";
|
||||
}
|
||||
})().then(console.log);
|
||||
console.log("moo");
|
||||
}
|
||||
expect_stdout: [
|
||||
"foo",
|
||||
"bar",
|
||||
"moo",
|
||||
"baz",
|
||||
]
|
||||
node_version: ">=8"
|
||||
}
|
||||
|
||||
issue_4972_3: {
|
||||
options = {
|
||||
awaits: true,
|
||||
side_effects: true,
|
||||
}
|
||||
input: {
|
||||
console.log("foo");
|
||||
try {
|
||||
(async function() {
|
||||
return await "bar";
|
||||
})().then(console.log);
|
||||
} finally {
|
||||
console.log("baz");
|
||||
}
|
||||
console.log("moo");
|
||||
}
|
||||
expect: {
|
||||
console.log("foo");
|
||||
try {
|
||||
(async function() {
|
||||
return "bar";
|
||||
})().then(console.log);
|
||||
} finally {
|
||||
console.log("baz");
|
||||
}
|
||||
console.log("moo");
|
||||
}
|
||||
expect_stdout: [
|
||||
"foo",
|
||||
"baz",
|
||||
"moo",
|
||||
"bar",
|
||||
]
|
||||
node_version: ">=8"
|
||||
}
|
||||
|
||||
issue_4974: {
|
||||
options = {
|
||||
awaits: true,
|
||||
side_effects: true,
|
||||
}
|
||||
input: {
|
||||
(async function f() {
|
||||
return 42 in f();
|
||||
})();
|
||||
console.log("PASS");
|
||||
}
|
||||
expect: {
|
||||
(async function f() {
|
||||
return 42 in f();
|
||||
})();
|
||||
console.log("PASS");
|
||||
}
|
||||
expect_stdout: true
|
||||
node_version: ">=8"
|
||||
}
|
||||
|
||||
issue_4975: {
|
||||
options = {
|
||||
awaits: true,
|
||||
side_effects: true,
|
||||
}
|
||||
input: {
|
||||
(async function f(a) {
|
||||
try {
|
||||
if (a) console.log(typeof f());
|
||||
} catch (e) {}
|
||||
})(42);
|
||||
}
|
||||
expect: {
|
||||
(async function f(a) {
|
||||
try {
|
||||
if (a) console.log(typeof f());
|
||||
} catch (e) {}
|
||||
})(42);
|
||||
}
|
||||
expect_stdout: "object"
|
||||
node_version: ">=8"
|
||||
}
|
||||
|
||||
issue_4987: {
|
||||
options = {
|
||||
awaits: true,
|
||||
side_effects: true,
|
||||
}
|
||||
input: {
|
||||
(async function() {
|
||||
try {
|
||||
await 42;
|
||||
} finally {
|
||||
console.log("foo");
|
||||
}
|
||||
})();
|
||||
console.log("bar");
|
||||
}
|
||||
expect: {
|
||||
(async function() {
|
||||
try {
|
||||
await 0;
|
||||
} finally {
|
||||
console.log("foo");
|
||||
}
|
||||
})();
|
||||
console.log("bar");
|
||||
}
|
||||
expect_stdout: [
|
||||
"bar",
|
||||
"foo",
|
||||
]
|
||||
node_version: ">=8"
|
||||
}
|
||||
|
||||
issue_5001: {
|
||||
options = {
|
||||
awaits: true,
|
||||
inline: true,
|
||||
side_effects: true,
|
||||
}
|
||||
input: {
|
||||
var a = 0;
|
||||
(async function() {
|
||||
a++ | await 42;
|
||||
})();
|
||||
console.log(a ? "PASS" : "FAIL");
|
||||
}
|
||||
expect: {
|
||||
var a = 0;
|
||||
a++;
|
||||
console.log(a ? "PASS" : "FAIL");
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=8"
|
||||
}
|
||||
|
||||
issue_5019_1: {
|
||||
options = {
|
||||
dead_code: true,
|
||||
}
|
||||
input: {
|
||||
(function(a) {
|
||||
(async function() {
|
||||
await 42;
|
||||
console.log(a);
|
||||
})();
|
||||
a = "PASS";
|
||||
})("FAIL");
|
||||
}
|
||||
expect: {
|
||||
(function(a) {
|
||||
(async function() {
|
||||
await 42;
|
||||
console.log(a);
|
||||
})();
|
||||
a = "PASS";
|
||||
})("FAIL");
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=8"
|
||||
}
|
||||
|
||||
issue_5019_2: {
|
||||
options = {
|
||||
dead_code: true,
|
||||
}
|
||||
input: {
|
||||
console.log("sync", function(a) {
|
||||
(async function() {
|
||||
console.log(await "async", a);
|
||||
})();
|
||||
return a = "PASS";
|
||||
}("FAIL"));
|
||||
}
|
||||
expect: {
|
||||
console.log("sync", function(a) {
|
||||
(async function() {
|
||||
console.log(await "async", a);
|
||||
})();
|
||||
return a = "PASS";
|
||||
}("FAIL"));
|
||||
}
|
||||
expect_stdout: [
|
||||
"sync PASS",
|
||||
"async PASS",
|
||||
]
|
||||
node_version: ">=8"
|
||||
}
|
||||
|
||||
issue_5019_3: {
|
||||
options = {
|
||||
inline: true,
|
||||
toplevel: true,
|
||||
}
|
||||
input: {
|
||||
for (var i in "foo") {
|
||||
(function(a) {
|
||||
(async function() {
|
||||
console.log(await "async", a);
|
||||
})();
|
||||
})(i);
|
||||
console.log("sync", i);
|
||||
}
|
||||
}
|
||||
expect: {
|
||||
for (var i in "foo") {
|
||||
(function(a) {
|
||||
(async function() {
|
||||
console.log(await "async", a);
|
||||
})();
|
||||
})(i);
|
||||
console.log("sync", i);
|
||||
}
|
||||
}
|
||||
expect_stdout: [
|
||||
"sync 0",
|
||||
"sync 1",
|
||||
"sync 2",
|
||||
"async 0",
|
||||
"async 1",
|
||||
"async 2",
|
||||
]
|
||||
node_version: ">=8"
|
||||
}
|
||||
|
||||
issue_5023_1: {
|
||||
options = {
|
||||
awaits: true,
|
||||
reduce_vars: true,
|
||||
side_effects: true,
|
||||
}
|
||||
input: {
|
||||
(async function() {
|
||||
let a = a;
|
||||
})();
|
||||
console.log("PASS");
|
||||
}
|
||||
expect: {
|
||||
(async function() {
|
||||
let a = a;
|
||||
})();
|
||||
console.log("PASS");
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=8"
|
||||
}
|
||||
|
||||
issue_5023_2: {
|
||||
options = {
|
||||
awaits: true,
|
||||
reduce_vars: true,
|
||||
side_effects: true,
|
||||
}
|
||||
input: {
|
||||
(async function() {
|
||||
let a;
|
||||
a = a;
|
||||
})();
|
||||
console.log("PASS");
|
||||
}
|
||||
expect: {
|
||||
(function() {
|
||||
let a;
|
||||
a = a;
|
||||
})();
|
||||
console.log("PASS");
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=8"
|
||||
}
|
||||
|
||||
issue_5032_normal: {
|
||||
options = {
|
||||
merge_vars: true,
|
||||
webkit: false,
|
||||
}
|
||||
input: {
|
||||
function log(value) {
|
||||
console.log(value);
|
||||
return value;
|
||||
}
|
||||
async function f(a) {
|
||||
var b = log(a), c = b;
|
||||
log(b);
|
||||
log(c);
|
||||
}
|
||||
f("PASS");
|
||||
}
|
||||
expect: {
|
||||
function log(value) {
|
||||
console.log(value);
|
||||
return value;
|
||||
}
|
||||
async function f(c) {
|
||||
var b = log(c), c = b;
|
||||
log(b);
|
||||
log(c);
|
||||
}
|
||||
f("PASS");
|
||||
}
|
||||
expect_stdout: [
|
||||
"PASS",
|
||||
"PASS",
|
||||
"PASS",
|
||||
]
|
||||
node_version: ">=8"
|
||||
}
|
||||
|
||||
issue_5032_webkit: {
|
||||
options = {
|
||||
merge_vars: true,
|
||||
webkit: true,
|
||||
}
|
||||
input: {
|
||||
function log(value) {
|
||||
console.log(value);
|
||||
return value;
|
||||
}
|
||||
async function f(a) {
|
||||
var b = log(a), c = b;
|
||||
log(b);
|
||||
log(c);
|
||||
}
|
||||
f("PASS");
|
||||
}
|
||||
expect: {
|
||||
function log(value) {
|
||||
console.log(value);
|
||||
return value;
|
||||
}
|
||||
async function f(a) {
|
||||
var b = log(a), c = b;
|
||||
log(b);
|
||||
log(c);
|
||||
}
|
||||
f("PASS");
|
||||
}
|
||||
expect_stdout: [
|
||||
"PASS",
|
||||
"PASS",
|
||||
"PASS",
|
||||
]
|
||||
node_version: ">=8"
|
||||
}
|
||||
|
||||
issue_5034: {
|
||||
options = {
|
||||
functions: true,
|
||||
reduce_vars: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
(function() {
|
||||
var await = function f() {
|
||||
return async function() {
|
||||
return f;
|
||||
};
|
||||
};
|
||||
await()().then(function(value) {
|
||||
console.log(value === await ? "PASS" : "FAIL");
|
||||
});
|
||||
})();
|
||||
}
|
||||
expect: {
|
||||
(function() {
|
||||
var await = function f() {
|
||||
return async function() {
|
||||
return f;
|
||||
};
|
||||
};
|
||||
await()().then(function(value) {
|
||||
console.log(value === await ? "PASS" : "FAIL");
|
||||
});
|
||||
})();
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=8"
|
||||
}
|
||||
|
||||
issue_5070: {
|
||||
options = {
|
||||
awaits: true,
|
||||
side_effects: true,
|
||||
}
|
||||
input: {
|
||||
(async function() {
|
||||
try {
|
||||
for await (var a of console.log("PASS"));
|
||||
} catch (e) {}
|
||||
})();
|
||||
}
|
||||
expect: {
|
||||
(async function() {
|
||||
try {
|
||||
for await (var a of console.log("PASS"));
|
||||
} catch (e) {}
|
||||
})();
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=10"
|
||||
}
|
||||
|
||||
@@ -60,3 +60,33 @@ issue_4590: {
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=10"
|
||||
}
|
||||
|
||||
issue_4801: {
|
||||
options = {
|
||||
booleans: true,
|
||||
collapse_vars: true,
|
||||
reduce_vars: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
try {
|
||||
(function(a) {
|
||||
A = 42;
|
||||
a || A;
|
||||
})(!(0 == 42 >> 0o644n));
|
||||
} catch (e) {
|
||||
console.log("PASS");
|
||||
}
|
||||
}
|
||||
expect: {
|
||||
try {
|
||||
(function(a) {
|
||||
0 != (A = 42) >> 0o644n || A;
|
||||
})();
|
||||
} catch (e) {
|
||||
console.log("PASS");
|
||||
}
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=10"
|
||||
}
|
||||
|
||||
@@ -29,6 +29,404 @@ iife_boolean_context: {
|
||||
]
|
||||
}
|
||||
|
||||
de_morgan_1a: {
|
||||
options = {
|
||||
booleans: true,
|
||||
}
|
||||
input: {
|
||||
function f(a) {
|
||||
return a || a;
|
||||
}
|
||||
console.log(f(null), f(42));
|
||||
}
|
||||
expect: {
|
||||
function f(a) {
|
||||
return a;
|
||||
}
|
||||
console.log(f(null), f(42));
|
||||
}
|
||||
expect_stdout: "null 42"
|
||||
}
|
||||
|
||||
de_morgan_1b: {
|
||||
options = {
|
||||
booleans: true,
|
||||
}
|
||||
input: {
|
||||
function f(a) {
|
||||
return a && a;
|
||||
}
|
||||
console.log(f(null), f(42));
|
||||
}
|
||||
expect: {
|
||||
function f(a) {
|
||||
return a;
|
||||
}
|
||||
console.log(f(null), f(42));
|
||||
}
|
||||
expect_stdout: "null 42"
|
||||
}
|
||||
|
||||
de_morgan_1c: {
|
||||
options = {
|
||||
booleans: true,
|
||||
}
|
||||
input: {
|
||||
console.log(delete (NaN && NaN));
|
||||
}
|
||||
expect: {
|
||||
console.log(delete (0, NaN));
|
||||
}
|
||||
expect_stdout: "true"
|
||||
}
|
||||
|
||||
de_morgan_2a: {
|
||||
options = {
|
||||
booleans: true,
|
||||
conditionals: true,
|
||||
}
|
||||
input: {
|
||||
function f(a, b) {
|
||||
return a || (a || b);
|
||||
}
|
||||
console.log(f(null), f(null, {}));
|
||||
console.log(f(42), f(42, {}));
|
||||
}
|
||||
expect: {
|
||||
function f(a, b) {
|
||||
return a || b;
|
||||
}
|
||||
console.log(f(null), f(null, {}));
|
||||
console.log(f(42), f(42, {}));
|
||||
}
|
||||
expect_stdout: [
|
||||
"undefined {}",
|
||||
"42 42",
|
||||
]
|
||||
}
|
||||
|
||||
de_morgan_2b: {
|
||||
options = {
|
||||
booleans: true,
|
||||
evaluate: true,
|
||||
}
|
||||
input: {
|
||||
function f(a, b) {
|
||||
return a || (a && b);
|
||||
}
|
||||
console.log(f(null), f(null, {}));
|
||||
console.log(f(42), f(42, {}));
|
||||
}
|
||||
expect: {
|
||||
function f(a, b) {
|
||||
return a;
|
||||
}
|
||||
console.log(f(null), f(null, {}));
|
||||
console.log(f(42), f(42, {}));
|
||||
}
|
||||
expect_stdout: [
|
||||
"null null",
|
||||
"42 42",
|
||||
]
|
||||
}
|
||||
|
||||
de_morgan_2c: {
|
||||
options = {
|
||||
booleans: true,
|
||||
evaluate: true,
|
||||
}
|
||||
input: {
|
||||
function f(a, b) {
|
||||
return a && (a || b);
|
||||
}
|
||||
console.log(f(null), f(null, {}));
|
||||
console.log(f(42), f(42, {}));
|
||||
}
|
||||
expect: {
|
||||
function f(a, b) {
|
||||
return a;
|
||||
}
|
||||
console.log(f(null), f(null, {}));
|
||||
console.log(f(42), f(42, {}));
|
||||
}
|
||||
expect_stdout: [
|
||||
"null null",
|
||||
"42 42",
|
||||
]
|
||||
}
|
||||
|
||||
de_morgan_2d: {
|
||||
options = {
|
||||
booleans: true,
|
||||
evaluate: true,
|
||||
side_effects: true,
|
||||
}
|
||||
input: {
|
||||
function f(a, b) {
|
||||
return a && (a && b);
|
||||
}
|
||||
console.log(f(null), f(null, {}));
|
||||
console.log(f(42), f(42, {}));
|
||||
}
|
||||
expect: {
|
||||
function f(a, b) {
|
||||
return a && b;
|
||||
}
|
||||
console.log(f(null), f(null, {}));
|
||||
console.log(f(42), f(42, {}));
|
||||
}
|
||||
expect_stdout: [
|
||||
"null null",
|
||||
"undefined {}",
|
||||
]
|
||||
}
|
||||
|
||||
de_morgan_3a: {
|
||||
options = {
|
||||
booleans: true,
|
||||
conditionals: true,
|
||||
}
|
||||
input: {
|
||||
function f(a, b, c) {
|
||||
return a || ((a || b) || c);
|
||||
}
|
||||
console.log(f(null, false), f(null, false, {}), f(null, true), f(null, true, {}));
|
||||
console.log(f(42, false), f(42, false, {}), f(42, true), f(42, true, {}));
|
||||
}
|
||||
expect: {
|
||||
function f(a, b, c) {
|
||||
return a || b || c;
|
||||
}
|
||||
console.log(f(null, !1), f(null, !1, {}), f(null, !0), f(null, !0, {}));
|
||||
console.log(f(42, !1), f(42, !1, {}), f(42, !0), f(42, !0, {}));
|
||||
}
|
||||
expect_stdout: [
|
||||
"undefined {} true true",
|
||||
"42 42 42 42",
|
||||
]
|
||||
}
|
||||
|
||||
de_morgan_3b: {
|
||||
options = {
|
||||
booleans: true,
|
||||
evaluate: true,
|
||||
side_effects: true,
|
||||
}
|
||||
input: {
|
||||
function f(a, b, c) {
|
||||
return a || ((a || b) && c);
|
||||
}
|
||||
console.log(f(null, false), f(null, false, {}), f(null, true), f(null, true, {}));
|
||||
console.log(f(42, false), f(42, false, {}), f(42, true), f(42, true, {}));
|
||||
}
|
||||
expect: {
|
||||
function f(a, b, c) {
|
||||
return a || b && c;
|
||||
}
|
||||
console.log(f(null, !1), f(null, !1, {}), f(null, !0), f(null, !0, {}));
|
||||
console.log(f(42, !1), f(42, !1, {}), f(42, !0), f(42, !0, {}));
|
||||
}
|
||||
expect_stdout: [
|
||||
"false false undefined {}",
|
||||
"42 42 42 42",
|
||||
]
|
||||
}
|
||||
|
||||
de_morgan_3c: {
|
||||
options = {
|
||||
booleans: true,
|
||||
evaluate: true,
|
||||
side_effects: true,
|
||||
}
|
||||
input: {
|
||||
function f(a, b, c) {
|
||||
return a || ((a && b) || c);
|
||||
}
|
||||
console.log(f(null, false), f(null, false, {}), f(null, true), f(null, true, {}));
|
||||
console.log(f(42, false), f(42, false, {}), f(42, true), f(42, true, {}));
|
||||
}
|
||||
expect: {
|
||||
function f(a, b, c) {
|
||||
return a || c;
|
||||
}
|
||||
console.log(f(null, !1), f(null, !1, {}), f(null, !0), f(null, !0, {}));
|
||||
console.log(f(42, !1), f(42, !1, {}), f(42, !0), f(42, !0, {}));
|
||||
}
|
||||
expect_stdout: [
|
||||
"undefined {} undefined {}",
|
||||
"42 42 42 42",
|
||||
]
|
||||
}
|
||||
|
||||
de_morgan_3d: {
|
||||
options = {
|
||||
booleans: true,
|
||||
evaluate: true,
|
||||
}
|
||||
input: {
|
||||
function f(a, b, c) {
|
||||
return a || ((a && b) && c);
|
||||
}
|
||||
console.log(f(null, false), f(null, false, {}), f(null, true), f(null, true, {}));
|
||||
console.log(f(42, false), f(42, false, {}), f(42, true), f(42, true, {}));
|
||||
}
|
||||
expect: {
|
||||
function f(a, b, c) {
|
||||
return a;
|
||||
}
|
||||
console.log(f(null, !1), f(null, !1, {}), f(null, !0), f(null, !0, {}));
|
||||
console.log(f(42, !1), f(42, !1, {}), f(42, !0), f(42, !0, {}));
|
||||
}
|
||||
expect_stdout: [
|
||||
"null null null null",
|
||||
"42 42 42 42",
|
||||
]
|
||||
}
|
||||
|
||||
de_morgan_3e: {
|
||||
options = {
|
||||
booleans: true,
|
||||
evaluate: true,
|
||||
}
|
||||
input: {
|
||||
function f(a, b, c) {
|
||||
return a && ((a || b) || c);
|
||||
}
|
||||
console.log(f(null, false), f(null, false, {}), f(null, true), f(null, true, {}));
|
||||
console.log(f(42, false), f(42, false, {}), f(42, true), f(42, true, {}));
|
||||
}
|
||||
expect: {
|
||||
function f(a, b, c) {
|
||||
return a;
|
||||
}
|
||||
console.log(f(null, !1), f(null, !1, {}), f(null, !0), f(null, !0, {}));
|
||||
console.log(f(42, !1), f(42, !1, {}), f(42, !0), f(42, !0, {}));
|
||||
}
|
||||
expect_stdout: [
|
||||
"null null null null",
|
||||
"42 42 42 42",
|
||||
]
|
||||
}
|
||||
|
||||
de_morgan_3f: {
|
||||
options = {
|
||||
booleans: true,
|
||||
evaluate: true,
|
||||
side_effects: true,
|
||||
}
|
||||
input: {
|
||||
function f(a, b, c) {
|
||||
return a && ((a || b) && c);
|
||||
}
|
||||
console.log(f(null, false), f(null, false, {}), f(null, true), f(null, true, {}));
|
||||
console.log(f(42, false), f(42, false, {}), f(42, true), f(42, true, {}));
|
||||
}
|
||||
expect: {
|
||||
function f(a, b, c) {
|
||||
return a && c;
|
||||
}
|
||||
console.log(f(null, !1), f(null, !1, {}), f(null, !0), f(null, !0, {}));
|
||||
console.log(f(42, !1), f(42, !1, {}), f(42, !0), f(42, !0, {}));
|
||||
}
|
||||
expect_stdout: [
|
||||
"null null null null",
|
||||
"undefined {} undefined {}",
|
||||
]
|
||||
}
|
||||
|
||||
de_morgan_3g: {
|
||||
options = {
|
||||
booleans: true,
|
||||
evaluate: true,
|
||||
side_effects: true,
|
||||
}
|
||||
input: {
|
||||
function f(a, b, c) {
|
||||
return a && ((a && b) || c);
|
||||
}
|
||||
console.log(f(null, false), f(null, false, {}), f(null, true), f(null, true, {}));
|
||||
console.log(f(42, false), f(42, false, {}), f(42, true), f(42, true, {}));
|
||||
}
|
||||
expect: {
|
||||
function f(a, b, c) {
|
||||
return a && (b || c);
|
||||
}
|
||||
console.log(f(null, !1), f(null, !1, {}), f(null, !0), f(null, !0, {}));
|
||||
console.log(f(42, !1), f(42, !1, {}), f(42, !0), f(42, !0, {}));
|
||||
}
|
||||
expect_stdout: [
|
||||
"null null null null",
|
||||
"undefined {} true true",
|
||||
]
|
||||
}
|
||||
|
||||
de_morgan_3h: {
|
||||
options = {
|
||||
booleans: true,
|
||||
conditionals: true,
|
||||
}
|
||||
input: {
|
||||
function f(a, b, c) {
|
||||
return a && ((a && b) && c);
|
||||
}
|
||||
console.log(f(null, false), f(null, false, {}), f(null, true), f(null, true, {}));
|
||||
console.log(f(42, false), f(42, false, {}), f(42, true), f(42, true, {}));
|
||||
}
|
||||
expect: {
|
||||
function f(a, b, c) {
|
||||
return a && b && c;
|
||||
}
|
||||
console.log(f(null, !1), f(null, !1, {}), f(null, !0), f(null, !0, {}));
|
||||
console.log(f(42, !1), f(42, !1, {}), f(42, !0), f(42, !0, {}));
|
||||
}
|
||||
expect_stdout: [
|
||||
"null null null null",
|
||||
"false false undefined {}",
|
||||
]
|
||||
}
|
||||
|
||||
conditional_chain: {
|
||||
options = {
|
||||
booleans: true,
|
||||
conditionals: true,
|
||||
evaluate: true,
|
||||
}
|
||||
input: {
|
||||
function f(a, b) {
|
||||
return a ? a : b ? b : 42;
|
||||
}
|
||||
console.log(f("PASS", "FAIL"));
|
||||
}
|
||||
expect: {
|
||||
function f(a, b) {
|
||||
return a || b || 42;
|
||||
}
|
||||
console.log(f("PASS", "FAIL"));
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
negated_if: {
|
||||
options = {
|
||||
booleans: true,
|
||||
conditionals: true,
|
||||
side_effects: true,
|
||||
}
|
||||
input: {
|
||||
console.log(function(a) {
|
||||
if (!a)
|
||||
return a ? "FAIL" : "PASS";
|
||||
}(!console));
|
||||
}
|
||||
expect: {
|
||||
console.log(function(a) {
|
||||
if (!a)
|
||||
return "PASS";
|
||||
}(!console));
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
issue_3465_1: {
|
||||
options = {
|
||||
booleans: true,
|
||||
@@ -181,3 +579,121 @@ issue_4374: {
|
||||
}
|
||||
expect_stdout: "0"
|
||||
}
|
||||
|
||||
issue_5028_1: {
|
||||
options = {
|
||||
booleans: true,
|
||||
conditionals: true,
|
||||
}
|
||||
input: {
|
||||
var a = 1;
|
||||
console.log(function() {
|
||||
return a-- ? a-- ? "FAIL 1" : "PASS" : "FAIL 2";
|
||||
}());
|
||||
}
|
||||
expect: {
|
||||
var a = 1;
|
||||
console.log(function() {
|
||||
return a-- ? a-- ? "FAIL 1" : "PASS" : "FAIL 2";
|
||||
}());
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
issue_5028_2: {
|
||||
options = {
|
||||
booleans: true,
|
||||
conditionals: true,
|
||||
dead_code: true,
|
||||
if_return: true,
|
||||
}
|
||||
input: {
|
||||
var a = 1;
|
||||
(function() {
|
||||
if (a--)
|
||||
if (a--)
|
||||
a = "FAIL";
|
||||
else
|
||||
return;
|
||||
})();
|
||||
console.log(a);
|
||||
}
|
||||
expect: {
|
||||
var a = 1;
|
||||
(function() {
|
||||
a-- && a-- && (a = "FAIL");
|
||||
})();
|
||||
console.log(a);
|
||||
}
|
||||
expect_stdout: "-1"
|
||||
}
|
||||
|
||||
issue_5028_3: {
|
||||
options = {
|
||||
booleans: true,
|
||||
conditionals: true,
|
||||
evaluate: true,
|
||||
if_return: true,
|
||||
}
|
||||
input: {
|
||||
var a = 1;
|
||||
(function() {
|
||||
if (a--)
|
||||
if (a--)
|
||||
a = "FAIL";
|
||||
else
|
||||
return;
|
||||
})();
|
||||
console.log(a);
|
||||
}
|
||||
expect: {
|
||||
var a = 1;
|
||||
(function() {
|
||||
a-- && a-- && (a = "FAIL");
|
||||
})();
|
||||
console.log(a);
|
||||
}
|
||||
expect_stdout: "-1"
|
||||
}
|
||||
|
||||
issue_5041_1: {
|
||||
options = {
|
||||
booleans: true,
|
||||
conditionals: true,
|
||||
}
|
||||
input: {
|
||||
var a = 42;
|
||||
if (a)
|
||||
if ([ a = null ])
|
||||
if (a)
|
||||
console.log("FAIL");
|
||||
else
|
||||
console.log("PASS");
|
||||
}
|
||||
expect: {
|
||||
var a = 42;
|
||||
a && [ a = null ] && (a ? console.log("FAIL") : console.log("PASS"));
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
issue_5041_2: {
|
||||
options = {
|
||||
booleans: true,
|
||||
conditionals: true,
|
||||
}
|
||||
input: {
|
||||
var a;
|
||||
if (!a)
|
||||
if (a = 42)
|
||||
if (a)
|
||||
console.log("PASS");
|
||||
else
|
||||
console.log("FAIL");
|
||||
}
|
||||
expect: {
|
||||
var a;
|
||||
a || (a = 42) && (a ? console.log("PASS") : console.log("FAIL"));
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
@@ -72,7 +72,7 @@ fields: {
|
||||
console.log(k, o[k]);
|
||||
console.log(o.q);
|
||||
}
|
||||
expect_exact: 'var o=new class A{"#p";static #p="PASS";async;get q(){return A.#p}[6*7]=console?"foo":"bar"};for(var k in o)console.log(k,o[k]);console.log(o.q);'
|
||||
expect_exact: 'var o=new class A{"#p";static#p="PASS";async;get q(){return A.#p}[6*7]=console?"foo":"bar"};for(var k in o)console.log(k,o[k]);console.log(o.q);'
|
||||
expect_stdout: [
|
||||
"42 foo",
|
||||
"#p undefined",
|
||||
@@ -136,7 +136,7 @@ private_methods: {
|
||||
}
|
||||
}().q.then(console.log);
|
||||
}
|
||||
expect_exact: "(new class A{static*#f(){yield 3*A.#p}async #g(){for(var a of A.#f())return a*await 2}static get #p(){return 7}get q(){return this.#g()}}).q.then(console.log);"
|
||||
expect_exact: "(new class A{static*#f(){yield 3*A.#p}async#g(){for(var a of A.#f())return a*await 2}static get#p(){return 7}get q(){return this.#g()}}).q.then(console.log);"
|
||||
expect_stdout: "42"
|
||||
node_version: ">=14.6"
|
||||
}
|
||||
@@ -260,6 +260,9 @@ block_scoped: {
|
||||
expect: {
|
||||
"use strict";
|
||||
0;
|
||||
{
|
||||
class A {}
|
||||
}
|
||||
if (console) {
|
||||
class B {}
|
||||
}
|
||||
@@ -269,6 +272,38 @@ block_scoped: {
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
retain_declaration: {
|
||||
options = {
|
||||
dead_code: true,
|
||||
}
|
||||
input: {
|
||||
"use strict";
|
||||
var a = "FAIL";
|
||||
try {
|
||||
console.log(function() {
|
||||
return a;
|
||||
class a {}
|
||||
}());
|
||||
} catch (e) {
|
||||
console.log("PASS");
|
||||
}
|
||||
}
|
||||
expect: {
|
||||
"use strict";
|
||||
var a = "FAIL";
|
||||
try {
|
||||
console.log(function() {
|
||||
return a;
|
||||
class a {}
|
||||
}());
|
||||
} catch (e) {
|
||||
console.log("PASS");
|
||||
}
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
drop_extends: {
|
||||
options = {
|
||||
inline: true,
|
||||
@@ -583,6 +618,31 @@ single_use_6: {
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
single_use_7: {
|
||||
options = {
|
||||
passes: 2,
|
||||
reduce_vars: true,
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
"use strict";
|
||||
class A {
|
||||
static foo() {}
|
||||
}
|
||||
var a = "foo" in A;
|
||||
console.log(a);
|
||||
}
|
||||
expect: {
|
||||
"use strict";
|
||||
console.log("foo" in class {
|
||||
static foo() {}
|
||||
});
|
||||
}
|
||||
expect_stdout: "true"
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
collapse_non_strict: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
@@ -607,6 +667,7 @@ collapse_non_strict: {
|
||||
collapse_rhs: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
unsafe: true,
|
||||
}
|
||||
input: {
|
||||
"use strict";
|
||||
@@ -654,6 +715,32 @@ collapse_rhs_static: {
|
||||
node_version: ">=12"
|
||||
}
|
||||
|
||||
self_comparison: {
|
||||
options = {
|
||||
booleans: true,
|
||||
comparisons: true,
|
||||
reduce_vars: true,
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
"use strict";
|
||||
class A {}
|
||||
console.log(A == A, A != A);
|
||||
console.log(A === A, A !== A);
|
||||
}
|
||||
expect: {
|
||||
"use strict";
|
||||
console.log(!0, !1);
|
||||
console.log(!0, !1);
|
||||
}
|
||||
expect_stdout: [
|
||||
"true false",
|
||||
"true false",
|
||||
]
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
property_side_effects: {
|
||||
options = {
|
||||
inline: true,
|
||||
@@ -725,8 +812,8 @@ unused_await: {
|
||||
(() => console.log(await))();
|
||||
})();
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=12"
|
||||
expect_stdout: true
|
||||
node_version: ">=12 <16"
|
||||
}
|
||||
|
||||
computed_key_side_effects: {
|
||||
@@ -779,6 +866,27 @@ computed_key_generator: {
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
keep_fnames: {
|
||||
options = {
|
||||
keep_fnames: true,
|
||||
toplevel: true,
|
||||
}
|
||||
mangle = {
|
||||
keep_fnames: true,
|
||||
toplevel: true,
|
||||
}
|
||||
input: {
|
||||
"use strict";
|
||||
class Foo {}
|
||||
console.log(Foo.name, class Bar {}.name);
|
||||
}
|
||||
expect: {
|
||||
"use strict";
|
||||
class Foo {}
|
||||
console.log(Foo.name, class Bar {}.name);
|
||||
}
|
||||
}
|
||||
|
||||
issue_805_1: {
|
||||
options = {
|
||||
inline: true,
|
||||
@@ -865,9 +973,9 @@ issue_4681: {
|
||||
}
|
||||
expect: {
|
||||
console.log(function(a) {
|
||||
class A {
|
||||
(class {
|
||||
static p = a = this;
|
||||
}
|
||||
});
|
||||
return typeof a;
|
||||
}());
|
||||
}
|
||||
@@ -1049,7 +1157,7 @@ issue_4705: {
|
||||
|
||||
issue_4720: {
|
||||
options = {
|
||||
ie8: true,
|
||||
ie: true,
|
||||
reduce_vars: true,
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
@@ -1275,3 +1383,686 @@ issue_4756: {
|
||||
]
|
||||
node_version: ">=12"
|
||||
}
|
||||
|
||||
issue_4821_1: {
|
||||
options = {
|
||||
reduce_vars: true,
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var a;
|
||||
class A {
|
||||
static p = void (a = this);
|
||||
}
|
||||
console.log(typeof a);
|
||||
}
|
||||
expect: {
|
||||
var a;
|
||||
(class {
|
||||
static p = void (a = this);
|
||||
});
|
||||
console.log(typeof a);
|
||||
}
|
||||
expect_stdout: "function"
|
||||
node_version: ">=12"
|
||||
}
|
||||
|
||||
issue_4821_2: {
|
||||
options = {
|
||||
side_effects: true,
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var a;
|
||||
class A {
|
||||
static p = void (a = this);
|
||||
}
|
||||
console.log(typeof a);
|
||||
}
|
||||
expect: {
|
||||
var a;
|
||||
(class {
|
||||
static p = void (a = this);
|
||||
});
|
||||
console.log(typeof a);
|
||||
}
|
||||
expect_stdout: "function"
|
||||
node_version: ">=12"
|
||||
}
|
||||
|
||||
issue_4829_1: {
|
||||
options = {
|
||||
properties: true,
|
||||
}
|
||||
input: {
|
||||
"use strict";
|
||||
try {
|
||||
class A extends { f(){} }.f {}
|
||||
} catch (e) {
|
||||
console.log("PASS");
|
||||
}
|
||||
}
|
||||
expect: {
|
||||
"use strict";
|
||||
try {
|
||||
class A extends [ () => {} ][0] {}
|
||||
} catch (e) {
|
||||
console.log("PASS");
|
||||
}
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
issue_4829_2: {
|
||||
options = {
|
||||
properties: true,
|
||||
}
|
||||
input: {
|
||||
"use strict";
|
||||
try {
|
||||
class A extends {
|
||||
f() {
|
||||
return arguments;
|
||||
},
|
||||
}.f {}
|
||||
} catch (e) {
|
||||
console.log("PASS");
|
||||
}
|
||||
}
|
||||
expect: {
|
||||
"use strict";
|
||||
try {
|
||||
class A extends {
|
||||
f() {
|
||||
return arguments;
|
||||
},
|
||||
}.f {}
|
||||
} catch (e) {
|
||||
console.log("PASS");
|
||||
}
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
mangle_properties: {
|
||||
mangle = {
|
||||
properties: {
|
||||
keep_quoted: true,
|
||||
},
|
||||
}
|
||||
input: {
|
||||
class A {
|
||||
static #P = "PASS";
|
||||
static get Q() {
|
||||
return this.#P;
|
||||
}
|
||||
#p(n) {
|
||||
return (this["q"] = n) * this.r;
|
||||
}
|
||||
set q(v) {
|
||||
this.r = v + 1;
|
||||
}
|
||||
r = this.#p(6);
|
||||
}
|
||||
console.log(A.Q, new A().r);
|
||||
}
|
||||
expect: {
|
||||
class A {
|
||||
static #t = "PASS";
|
||||
static get s() {
|
||||
return this.#t;
|
||||
}
|
||||
#i(t) {
|
||||
return (this["q"] = t) * this.e;
|
||||
}
|
||||
set q(t) {
|
||||
this.e = t + 1;
|
||||
}
|
||||
e = this.#i(6);
|
||||
}
|
||||
console.log(A.s, new A().e);
|
||||
}
|
||||
expect_stdout: "PASS 42"
|
||||
node_version: ">=14.6"
|
||||
}
|
||||
|
||||
issue_4848: {
|
||||
options = {
|
||||
if_return: true,
|
||||
}
|
||||
input: {
|
||||
"use strict";
|
||||
function f(a) {
|
||||
a(function() {
|
||||
new A();
|
||||
});
|
||||
if (!console)
|
||||
return;
|
||||
class A {
|
||||
constructor() {
|
||||
console.log("PASS");
|
||||
}
|
||||
}
|
||||
}
|
||||
var g;
|
||||
f(function(h) {
|
||||
g = h;
|
||||
});
|
||||
g();
|
||||
}
|
||||
expect: {
|
||||
"use strict";
|
||||
function f(a) {
|
||||
a(function() {
|
||||
new A();
|
||||
});
|
||||
if (!console)
|
||||
return;
|
||||
class A {
|
||||
constructor() {
|
||||
console.log("PASS");
|
||||
}
|
||||
}
|
||||
}
|
||||
var g;
|
||||
f(function(h) {
|
||||
g = h;
|
||||
});
|
||||
g();
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
drop_unused_self_reference: {
|
||||
options = {
|
||||
pure_getters: "strict",
|
||||
reduce_vars: true,
|
||||
sequences: true,
|
||||
side_effects: true,
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
"use strict";
|
||||
class A {}
|
||||
(A.p = A).q = console.log("PASS");
|
||||
}
|
||||
expect: {
|
||||
"use strict";
|
||||
console.log("PASS");
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
issue_4951_1: {
|
||||
input: {
|
||||
class A {
|
||||
static#p = console.log("PASS");
|
||||
}
|
||||
}
|
||||
expect_exact: 'class A{static#p=console.log("PASS")}'
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=12"
|
||||
}
|
||||
|
||||
issue_4951_2: {
|
||||
input: {
|
||||
new class {
|
||||
constructor() {
|
||||
this.#f().then(console.log);
|
||||
}
|
||||
async#f() {
|
||||
return await "PASS";
|
||||
}
|
||||
}();
|
||||
}
|
||||
expect_exact: 'new class{constructor(){this.#f().then(console.log)}async#f(){return await"PASS"}};'
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=14.6"
|
||||
}
|
||||
|
||||
issue_4962_1: {
|
||||
options = {
|
||||
ie: true,
|
||||
inline: true,
|
||||
reduce_vars: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
(function() {
|
||||
function f() {
|
||||
while (console.log(typeof g));
|
||||
}
|
||||
class A {
|
||||
static p = f();
|
||||
}
|
||||
})(function g() {});
|
||||
}
|
||||
expect: {
|
||||
(function g() {}),
|
||||
void function() {
|
||||
while (console.log(typeof g));
|
||||
}();
|
||||
}
|
||||
expect_stdout: "undefined"
|
||||
node_version: ">=12"
|
||||
}
|
||||
|
||||
issue_4962_2: {
|
||||
options = {
|
||||
ie: true,
|
||||
inline: true,
|
||||
reduce_vars: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
console.log(function f() {}(function g() {
|
||||
function h() {
|
||||
f;
|
||||
}
|
||||
class A {
|
||||
static p = h();
|
||||
}
|
||||
}, typeof g));
|
||||
}
|
||||
expect: {
|
||||
console.log(function f() {}(function g() {
|
||||
f;
|
||||
}));
|
||||
}
|
||||
expect_stdout: "undefined"
|
||||
node_version: ">=12"
|
||||
}
|
||||
|
||||
issue_4982_1: {
|
||||
options = {
|
||||
dead_code: true,
|
||||
}
|
||||
input: {
|
||||
"use strict";
|
||||
try {} catch (e) {
|
||||
class A extends 42 {}
|
||||
}
|
||||
console.log("PASS");
|
||||
}
|
||||
expect: {
|
||||
"use strict";
|
||||
{
|
||||
class A {}
|
||||
}
|
||||
console.log("PASS");
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
issue_4982_2: {
|
||||
options = {
|
||||
dead_code: true,
|
||||
}
|
||||
input: {
|
||||
var a = "PASS";
|
||||
try {} catch (e) {
|
||||
class A {
|
||||
static p = a = "FAIL";
|
||||
}
|
||||
}
|
||||
console.log(a);
|
||||
}
|
||||
expect: {
|
||||
var a = "PASS";
|
||||
{
|
||||
class A {}
|
||||
}
|
||||
console.log(a);
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=12"
|
||||
}
|
||||
|
||||
issue_4992: {
|
||||
options = {
|
||||
reduce_vars: true,
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
class A {
|
||||
static P = this;
|
||||
get p() {}
|
||||
}
|
||||
console.log(typeof A.P);
|
||||
}
|
||||
expect: {
|
||||
console.log(typeof class {
|
||||
static P = this;
|
||||
get p() {}
|
||||
}.P);
|
||||
}
|
||||
expect_stdout: "function"
|
||||
node_version: ">=12"
|
||||
}
|
||||
|
||||
issue_4996_1: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
reduce_vars: true,
|
||||
toplevel: true,
|
||||
}
|
||||
input: {
|
||||
var a = 1;
|
||||
console.log(new class A {
|
||||
p = a-- && new A();
|
||||
}().p.p);
|
||||
}
|
||||
expect: {
|
||||
var a = 1;
|
||||
console.log(new class A {
|
||||
p = a-- && new A();
|
||||
}().p.p);
|
||||
}
|
||||
expect_stdout: "0"
|
||||
node_version: ">=12"
|
||||
}
|
||||
|
||||
issue_4996_2: {
|
||||
options = {
|
||||
reduce_vars: true,
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var a = 1;
|
||||
console.log(new class A {
|
||||
p = a-- && new A();
|
||||
}().p.p);
|
||||
}
|
||||
expect: {
|
||||
var a = 1;
|
||||
console.log(new class A {
|
||||
p = a-- && new A();
|
||||
}().p.p);
|
||||
}
|
||||
expect_stdout: "0"
|
||||
node_version: ">=12"
|
||||
}
|
||||
|
||||
issue_5015_1: {
|
||||
options = {
|
||||
side_effects: true,
|
||||
}
|
||||
input: {
|
||||
"use strict";
|
||||
var a;
|
||||
try {
|
||||
(class a {
|
||||
[a]() {}
|
||||
});
|
||||
} catch (e) {
|
||||
console.log("PASS");
|
||||
}
|
||||
}
|
||||
expect: {
|
||||
"use strict";
|
||||
var a;
|
||||
try {
|
||||
(class a {
|
||||
[a]() {}
|
||||
});
|
||||
} catch (e) {
|
||||
console.log("PASS");
|
||||
}
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
issue_5015_2: {
|
||||
options = {
|
||||
reduce_vars: true,
|
||||
side_effects: true,
|
||||
toplevel: true,
|
||||
}
|
||||
input: {
|
||||
"use strict";
|
||||
try {
|
||||
new class A {
|
||||
[(A, 42)]() {}
|
||||
}();
|
||||
} catch (e) {
|
||||
console.log("PASS");
|
||||
}
|
||||
}
|
||||
expect: {
|
||||
"use strict";
|
||||
try {
|
||||
new class A {
|
||||
[(A, 42)]() {}
|
||||
}();
|
||||
} catch (e) {
|
||||
console.log("PASS");
|
||||
}
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
issue_5015_3: {
|
||||
options = {
|
||||
reduce_vars: true,
|
||||
side_effects: true,
|
||||
toplevel: true,
|
||||
}
|
||||
input: {
|
||||
"use strict";
|
||||
(class A {
|
||||
static f() {
|
||||
return A;
|
||||
}
|
||||
});
|
||||
console.log("PASS");
|
||||
}
|
||||
expect: {
|
||||
"use strict";
|
||||
console.log("PASS");
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
issue_5053_1: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
reduce_vars: true,
|
||||
toplevel: true,
|
||||
}
|
||||
input: {
|
||||
"use strict";
|
||||
try {
|
||||
console.log(new class A {
|
||||
constructor() {
|
||||
A = 42;
|
||||
}
|
||||
}());
|
||||
} catch (e) {
|
||||
console.log("PASS");
|
||||
}
|
||||
}
|
||||
expect: {
|
||||
"use strict";
|
||||
try {
|
||||
console.log(new class A {
|
||||
constructor() {
|
||||
A = 42;
|
||||
}
|
||||
}());
|
||||
} catch (e) {
|
||||
console.log("PASS");
|
||||
}
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
issue_5053_2: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
reduce_vars: true,
|
||||
toplevel: true,
|
||||
}
|
||||
input: {
|
||||
"use strict";
|
||||
try {
|
||||
console.log(new class A {
|
||||
f() {
|
||||
A = 42;
|
||||
}
|
||||
}().f());
|
||||
} catch (e) {
|
||||
console.log("PASS");
|
||||
}
|
||||
}
|
||||
expect: {
|
||||
"use strict";
|
||||
try {
|
||||
console.log(new class A {
|
||||
f() {
|
||||
A = 42;
|
||||
}
|
||||
}().f());
|
||||
} catch (e) {
|
||||
console.log("PASS");
|
||||
}
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
issue_5053_3: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
reduce_vars: true,
|
||||
toplevel: true,
|
||||
}
|
||||
input: {
|
||||
try {
|
||||
console.log(new class A {
|
||||
p = A = 42;
|
||||
}().p);
|
||||
} catch (e) {
|
||||
console.log("PASS");
|
||||
}
|
||||
}
|
||||
expect: {
|
||||
try {
|
||||
console.log(new class A {
|
||||
p = A = 42;
|
||||
}().p);
|
||||
} catch (e) {
|
||||
console.log("PASS");
|
||||
}
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=12"
|
||||
}
|
||||
|
||||
issue_5053_4: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
reduce_vars: true,
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
"use strict";
|
||||
class A {
|
||||
constructor() {
|
||||
A = 42;
|
||||
}
|
||||
}
|
||||
try {
|
||||
console.log(new A());
|
||||
} catch (e) {
|
||||
console.log("PASS");
|
||||
}
|
||||
}
|
||||
expect: {
|
||||
"use strict";
|
||||
class A {
|
||||
constructor() {
|
||||
A = 42;
|
||||
}
|
||||
}
|
||||
try {
|
||||
console.log(new A());
|
||||
} catch (e) {
|
||||
console.log("PASS");
|
||||
}
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
issue_5082_1: {
|
||||
options = {
|
||||
inline: true,
|
||||
reduce_vars: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
(function() {
|
||||
class A {
|
||||
p = console.log("PASS");
|
||||
q() {}
|
||||
}
|
||||
class B {
|
||||
static P = new A();
|
||||
}
|
||||
})();
|
||||
}
|
||||
expect: {
|
||||
(function() {
|
||||
class A {
|
||||
p = console.log("PASS");
|
||||
q() {}
|
||||
}
|
||||
new A();
|
||||
})();
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=12"
|
||||
}
|
||||
|
||||
issue_5082_2: {
|
||||
options = {
|
||||
inline: true,
|
||||
passes: 2,
|
||||
reduce_vars: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
(function() {
|
||||
class A {
|
||||
p = console.log("PASS");
|
||||
q() {}
|
||||
}
|
||||
class B {
|
||||
static P = new A();
|
||||
}
|
||||
})();
|
||||
}
|
||||
expect: {
|
||||
void new class {
|
||||
p = console.log("PASS");
|
||||
q() {}
|
||||
}();
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=12"
|
||||
}
|
||||
|
||||
@@ -2264,6 +2264,7 @@ var_defs: {
|
||||
properties: true,
|
||||
sequences: true,
|
||||
side_effects: true,
|
||||
unsafe: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
@@ -2486,6 +2487,7 @@ issue_1858: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
pure_getters: true,
|
||||
reduce_vars: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
@@ -2868,7 +2870,7 @@ lvalues_def: {
|
||||
expect_stdout: true
|
||||
}
|
||||
|
||||
compound_assignment: {
|
||||
compound_assignment_1: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
}
|
||||
@@ -2887,9 +2889,27 @@ compound_assignment: {
|
||||
expect_stdout: "4"
|
||||
}
|
||||
|
||||
compound_assignment_2: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
}
|
||||
input: {
|
||||
var a;
|
||||
a = 1;
|
||||
for (a += a + 2; console.log(a););
|
||||
}
|
||||
expect: {
|
||||
var a;
|
||||
a = 1;
|
||||
for (a += a + 2; console.log(a););
|
||||
}
|
||||
expect_stdout: "4"
|
||||
}
|
||||
|
||||
issue_2187_1: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
reduce_vars: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
@@ -4254,6 +4274,7 @@ issue_2436_14: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
reduce_vars: true,
|
||||
unsafe: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
@@ -4744,6 +4765,7 @@ unsafe_builtin: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
pure_getters: "strict",
|
||||
reduce_vars: true,
|
||||
unsafe: true,
|
||||
unused: true,
|
||||
}
|
||||
@@ -5986,7 +6008,7 @@ issue_3215_1: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
evaluate: true,
|
||||
ie8: false,
|
||||
ie: false,
|
||||
inline: true,
|
||||
passes: 2,
|
||||
side_effects: true,
|
||||
@@ -6008,7 +6030,7 @@ issue_3215_2: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
evaluate: true,
|
||||
ie8: true,
|
||||
ie: true,
|
||||
inline: true,
|
||||
passes: 2,
|
||||
side_effects: true,
|
||||
@@ -6033,7 +6055,7 @@ issue_3215_3: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
evaluate: true,
|
||||
ie8: false,
|
||||
ie: false,
|
||||
inline: true,
|
||||
passes: 2,
|
||||
side_effects: true,
|
||||
@@ -6056,7 +6078,7 @@ issue_3215_4: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
evaluate: true,
|
||||
ie8: true,
|
||||
ie: true,
|
||||
inline: true,
|
||||
passes: 2,
|
||||
side_effects: true,
|
||||
@@ -6948,6 +6970,7 @@ sequence_in_iife_2: {
|
||||
inline: true,
|
||||
passes: 2,
|
||||
side_effects: true,
|
||||
unsafe: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
@@ -6959,7 +6982,7 @@ sequence_in_iife_2: {
|
||||
}
|
||||
expect: {
|
||||
var a = "foo", b = 42;
|
||||
console.log(a, b = a);
|
||||
console.log(b = a, b);
|
||||
}
|
||||
expect_stdout: "foo foo"
|
||||
}
|
||||
@@ -6971,6 +6994,7 @@ sequence_in_iife_3: {
|
||||
passes: 2,
|
||||
side_effects: true,
|
||||
toplevel: true,
|
||||
unsafe: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
@@ -7075,6 +7099,7 @@ setter_side_effect: {
|
||||
substitution_assign: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
unsafe: true,
|
||||
}
|
||||
input: {
|
||||
function f1(a, b) {
|
||||
@@ -7095,8 +7120,7 @@ substitution_assign: {
|
||||
}
|
||||
expect: {
|
||||
function f1(a, b) {
|
||||
f1 = a;
|
||||
console.log(a, a);
|
||||
console.log(f1 = a, a);
|
||||
}
|
||||
function f2(a, b) {
|
||||
a = 1 + (b = a);
|
||||
@@ -7900,6 +7924,7 @@ issue_3744: {
|
||||
assign_value_def: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
reduce_vars: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
@@ -7940,6 +7965,7 @@ join_vars_value_def: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
join_vars: true,
|
||||
reduce_vars: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
@@ -7979,6 +8005,7 @@ join_vars_value_def: {
|
||||
var_value_def: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
reduce_vars: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
@@ -8016,6 +8043,7 @@ var_value_def: {
|
||||
mangleable_var: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
reduce_vars: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
@@ -8047,6 +8075,69 @@ mangleable_var: {
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
mangleable_assignment_1: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
reduce_vars: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var o = {
|
||||
p: function() {
|
||||
return 6;
|
||||
},
|
||||
};
|
||||
(function() {
|
||||
var a, b = a = o.p();
|
||||
console.log(a * (b / a + b));
|
||||
})();
|
||||
}
|
||||
expect: {
|
||||
var o = {
|
||||
p: function() {
|
||||
return 6;
|
||||
},
|
||||
};
|
||||
(function() {
|
||||
var a;
|
||||
a = o.p();
|
||||
console.log(a * (a / a + a));
|
||||
})();
|
||||
}
|
||||
expect_stdout: "42"
|
||||
}
|
||||
|
||||
mangleable_assignment_2: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
reduce_vars: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var o = {
|
||||
p: function() {
|
||||
return 6;
|
||||
},
|
||||
};
|
||||
(function(a, b) {
|
||||
b = a = o.p();
|
||||
console.log(a * (b / a + b));
|
||||
})();
|
||||
}
|
||||
expect: {
|
||||
var o = {
|
||||
p: function() {
|
||||
return 6;
|
||||
},
|
||||
};
|
||||
(function(a, b) {
|
||||
a = o.p();
|
||||
console.log(a * (a / a + a));
|
||||
})();
|
||||
}
|
||||
expect_stdout: "42"
|
||||
}
|
||||
|
||||
issue_3884_1: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
@@ -8482,7 +8573,7 @@ issue_4047_1: {
|
||||
expect: {
|
||||
var b = 1;
|
||||
var a;
|
||||
console.log((a = --b + ((a = 0) !== typeof A), +void ((a >>= 0) && console.log("PASS"))));
|
||||
console.log((a = --b + (0 !== typeof A), +void ((a >>= 0) && console.log("PASS"))));
|
||||
}
|
||||
expect_stdout: [
|
||||
"PASS",
|
||||
@@ -8772,7 +8863,7 @@ issue_4732_1: {
|
||||
expect: {
|
||||
var a = 0;
|
||||
(function(b) {
|
||||
(b = a++) && (b && console.log("PASS"));
|
||||
(b = a++) && console.log("PASS");
|
||||
})(a++);
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
@@ -8857,3 +8948,537 @@ dot_non_local: {
|
||||
}
|
||||
expect_stdout: "42"
|
||||
}
|
||||
|
||||
issue_4806: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
reduce_vars: true,
|
||||
toplevel: true,
|
||||
}
|
||||
input: {
|
||||
var a, o = {
|
||||
f: function() {
|
||||
console.log(this === o ? "FAIL" : "PASS");
|
||||
},
|
||||
};
|
||||
(a = 42, o.f)(42);
|
||||
}
|
||||
expect: {
|
||||
var a, o = {
|
||||
f: function() {
|
||||
console.log(this === o ? "FAIL" : "PASS");
|
||||
},
|
||||
};
|
||||
(0, o.f)(a = 42);
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
issue_4852: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
unsafe: true,
|
||||
}
|
||||
input: {
|
||||
var a = "PASS";
|
||||
(function(b) {
|
||||
switch (b = a) {
|
||||
case 42:
|
||||
try {
|
||||
console;
|
||||
} catch (b) {
|
||||
b.p;
|
||||
}
|
||||
case console.log(b):
|
||||
}
|
||||
})("FAIL");
|
||||
}
|
||||
expect: {
|
||||
var a = "PASS";
|
||||
(function(b) {
|
||||
switch (a) {
|
||||
case 42:
|
||||
try {
|
||||
console;
|
||||
} catch (b) {
|
||||
b.p;
|
||||
}
|
||||
case console.log(a):
|
||||
}
|
||||
})("FAIL");
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
issue_4865: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
}
|
||||
input: {
|
||||
var NaN;
|
||||
var a = NaN = "PASS";
|
||||
console.log(a, NaN);
|
||||
}
|
||||
expect: {
|
||||
var NaN;
|
||||
var a = NaN = "PASS";
|
||||
console.log(a, NaN);
|
||||
}
|
||||
expect_stdout: true
|
||||
}
|
||||
|
||||
issue_4868: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var a;
|
||||
(function(b) {
|
||||
console.log(b[0]);
|
||||
})(a = [ "PASS" ], a = [ "FAIL" ]);
|
||||
}
|
||||
expect: {
|
||||
var a;
|
||||
(function(b) {
|
||||
console.log(b[0]);
|
||||
})(a = [ "PASS" ], a = [ "FAIL" ]);
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
issue_4874: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
evaluate: true,
|
||||
reduce_vars: true,
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var a;
|
||||
a = null;
|
||||
(function(b) {
|
||||
for (var c in b = b && b[console.log("PASS")])
|
||||
console;
|
||||
})(a = 42);
|
||||
}
|
||||
expect: {
|
||||
var a;
|
||||
null;
|
||||
(function(b) {
|
||||
for (var c in a && a[console.log("PASS")])
|
||||
console;
|
||||
})(a = 42);
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
issue_4891: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
reduce_vars: true,
|
||||
toplevel: true,
|
||||
}
|
||||
input: {
|
||||
var a = 0, b;
|
||||
a++;
|
||||
console.log(b = a, b);
|
||||
b--;
|
||||
a.a += 0;
|
||||
console.log(b);
|
||||
}
|
||||
expect: {
|
||||
var a = 0, b;
|
||||
a++;
|
||||
console.log(a, b = a);
|
||||
b--;
|
||||
a.a += 0;
|
||||
console.log(b);
|
||||
}
|
||||
expect_stdout: [
|
||||
"1 1",
|
||||
"0",
|
||||
]
|
||||
}
|
||||
|
||||
issue_4895: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
toplevel: true,
|
||||
}
|
||||
input: {
|
||||
var a, b;
|
||||
(function f() {
|
||||
a = 42;
|
||||
})();
|
||||
console.log((b = a) || b, b += 0);
|
||||
}
|
||||
expect: {
|
||||
var a, b;
|
||||
(function f() {
|
||||
a = 42;
|
||||
})();
|
||||
console.log((b = a) || b, b += 0);
|
||||
}
|
||||
expect_stdout: "42 42"
|
||||
}
|
||||
|
||||
issue_4908: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
join_vars: true,
|
||||
reduce_vars: true,
|
||||
toplevel: true,
|
||||
}
|
||||
input: {
|
||||
var a = 0;
|
||||
var b;
|
||||
console || a++;
|
||||
var c = d = a, d = [ c && c, d += 42 ];
|
||||
console.log(d[1]);
|
||||
}
|
||||
expect: {
|
||||
var a = 0, b;
|
||||
console || a++;
|
||||
var c = a, d = [ (d = a) && d, d += 42 ];
|
||||
console.log(d[1]);
|
||||
}
|
||||
expect_stdout: "42"
|
||||
}
|
||||
|
||||
issue_4910: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
reduce_vars: true,
|
||||
toplevel: true,
|
||||
}
|
||||
input: {
|
||||
var a = "foo", b;
|
||||
var c = b = a;
|
||||
1 && c[a = "bar"];
|
||||
console.log(a, b);
|
||||
}
|
||||
expect: {
|
||||
var a = "foo", b;
|
||||
var c = b = a;
|
||||
1 && b[a = "bar"];
|
||||
console.log(a, b);
|
||||
}
|
||||
expect_stdout: "bar foo"
|
||||
}
|
||||
|
||||
issue_4914: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
pure_getters: "strict",
|
||||
}
|
||||
input: {
|
||||
console.log(typeof function f() {
|
||||
f.__proto__ = 42;
|
||||
return f.__proto__;
|
||||
}());
|
||||
}
|
||||
expect: {
|
||||
console.log(typeof function f() {
|
||||
f.__proto__ = 42;
|
||||
return f.__proto__;
|
||||
}());
|
||||
}
|
||||
expect_stdout: "function"
|
||||
}
|
||||
|
||||
issue_4918: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
}
|
||||
input: {
|
||||
var a = "FAIL";
|
||||
({
|
||||
get 42() {
|
||||
console.log(a);
|
||||
}
|
||||
}[a = "PASS", 42] += "PASS");
|
||||
}
|
||||
expect: {
|
||||
var a = "FAIL";
|
||||
({
|
||||
get 42() {
|
||||
console.log(a);
|
||||
}
|
||||
}[a = "PASS", 42] += "PASS");
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
issue_4920_1: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
toplevel: true,
|
||||
}
|
||||
input: {
|
||||
var a = "PASS", b;
|
||||
({
|
||||
get PASS() {
|
||||
a = "FAIL";
|
||||
},
|
||||
})[b = a];
|
||||
console.log(b);
|
||||
}
|
||||
expect: {
|
||||
var a = "PASS", b;
|
||||
({
|
||||
get PASS() {
|
||||
a = "FAIL";
|
||||
},
|
||||
})[b = a];
|
||||
console.log(b);
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
issue_4920_2: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
toplevel: true,
|
||||
}
|
||||
input: {
|
||||
var o = {
|
||||
get PASS() {
|
||||
a = "FAIL";
|
||||
},
|
||||
};
|
||||
var a = "PASS", b;
|
||||
o[b = a];
|
||||
console.log(b);
|
||||
}
|
||||
expect: {
|
||||
var o = {
|
||||
get PASS() {
|
||||
a = "FAIL";
|
||||
},
|
||||
};
|
||||
var a = "PASS", b;
|
||||
o[b = a];
|
||||
console.log(b);
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
issue_4920_3: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
toplevel: true,
|
||||
}
|
||||
input: {
|
||||
var log = console.log;
|
||||
var o = {
|
||||
get PASS() {
|
||||
a = "FAIL";
|
||||
},
|
||||
};
|
||||
var a = "PASS", b;
|
||||
o[b = a];
|
||||
log(b);
|
||||
}
|
||||
expect: {
|
||||
var log = console.log;
|
||||
var o = {
|
||||
get PASS() {
|
||||
a = "FAIL";
|
||||
},
|
||||
};
|
||||
var a = "PASS", b;
|
||||
o[b = a];
|
||||
log(b);
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
issue_4935: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
reduce_vars: true,
|
||||
toplevel: true,
|
||||
}
|
||||
input: {
|
||||
var a = 1;
|
||||
var b;
|
||||
var c = b = a;
|
||||
console || c(a++);
|
||||
--b;
|
||||
console.log(a, b);
|
||||
}
|
||||
expect: {
|
||||
var a = 1;
|
||||
var b;
|
||||
var c = b = a;
|
||||
console || a(a++);
|
||||
--b;
|
||||
console.log(a, b);
|
||||
}
|
||||
expect_stdout: "1 0"
|
||||
}
|
||||
|
||||
inline_throw: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
inline: true,
|
||||
keep_fargs: false,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
try {
|
||||
(function() {
|
||||
return function(a) {
|
||||
return function(b) {
|
||||
throw b;
|
||||
}(a);
|
||||
};
|
||||
})()("PASS");
|
||||
} catch (e) {
|
||||
console.log(e);
|
||||
}
|
||||
}
|
||||
expect: {
|
||||
try {
|
||||
(function(a) {
|
||||
return function() {
|
||||
throw a;
|
||||
}();
|
||||
})("PASS");
|
||||
} catch (e) {
|
||||
console.log(e);
|
||||
}
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
issue_4977_1: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
unsafe: true,
|
||||
}
|
||||
input: {
|
||||
var a = "FAIL";
|
||||
var o = {
|
||||
get p() {
|
||||
return a;
|
||||
}
|
||||
};
|
||||
a = "PASS";
|
||||
console.log(o.p, a);
|
||||
}
|
||||
expect: {
|
||||
var a = "FAIL";
|
||||
var o = {
|
||||
get p() {
|
||||
return a;
|
||||
}
|
||||
};
|
||||
a = "PASS";
|
||||
console.log(o.p, a);
|
||||
}
|
||||
expect_stdout: "PASS PASS"
|
||||
}
|
||||
|
||||
issue_4977_2: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
reduce_vars: true,
|
||||
unsafe: true,
|
||||
}
|
||||
input: {
|
||||
var a, o = {
|
||||
get p() {
|
||||
return a = "PASS";
|
||||
},
|
||||
};
|
||||
if (console) {
|
||||
var a = "FAIL";
|
||||
console.log(o.p, a);
|
||||
}
|
||||
}
|
||||
expect: {
|
||||
var a, o = {
|
||||
get p() {
|
||||
return a = "PASS";
|
||||
},
|
||||
};
|
||||
if (console) {
|
||||
var a = "FAIL";
|
||||
console.log(o.p, a);
|
||||
}
|
||||
}
|
||||
expect_stdout: "PASS PASS"
|
||||
}
|
||||
|
||||
issue_5112_1: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
}
|
||||
input: {
|
||||
console.log(function(a) {
|
||||
try {
|
||||
try {
|
||||
if (console + (a = "PASS", ""))
|
||||
return "FAIL 1";
|
||||
a.p;
|
||||
} catch (e) {}
|
||||
} finally {
|
||||
return a;
|
||||
}
|
||||
}("FAIL 2"));
|
||||
}
|
||||
expect: {
|
||||
console.log(function(a) {
|
||||
try {
|
||||
try {
|
||||
if (console + (a = "PASS", ""))
|
||||
return "FAIL 1";
|
||||
a.p;
|
||||
} catch (e) {}
|
||||
} finally {
|
||||
return a;
|
||||
}
|
||||
}("FAIL 2"));
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
issue_5112_2: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
}
|
||||
input: {
|
||||
console.log(function(a) {
|
||||
try {
|
||||
return function() {
|
||||
try {
|
||||
if (console + (a = "PASS", ""))
|
||||
return "FAIL 1";
|
||||
a.p;
|
||||
} catch (e) {}
|
||||
}();
|
||||
} finally {
|
||||
return a;
|
||||
}
|
||||
}("FAIL 2"));
|
||||
}
|
||||
expect: {
|
||||
console.log(function(a) {
|
||||
try {
|
||||
return function() {
|
||||
try {
|
||||
if (console + (a = "PASS", ""))
|
||||
return "FAIL 1";
|
||||
a.p;
|
||||
} catch (e) {}
|
||||
}();
|
||||
} finally {
|
||||
return a;
|
||||
}
|
||||
}("FAIL 2"));
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
@@ -822,6 +822,33 @@ cond_13: {
|
||||
}
|
||||
}
|
||||
|
||||
cond_14: {
|
||||
options = {
|
||||
booleans: true,
|
||||
conditionals: true,
|
||||
side_effects: true,
|
||||
}
|
||||
input: {
|
||||
function f(a) {
|
||||
if (a)
|
||||
if (a)
|
||||
console.log("PASS");
|
||||
else
|
||||
console.log("FAIL");
|
||||
}
|
||||
f(null);
|
||||
f(42);
|
||||
}
|
||||
expect: {
|
||||
function f(a) {
|
||||
a && console.log("PASS");
|
||||
}
|
||||
f(null);
|
||||
f(42);
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
ternary_boolean_consequent: {
|
||||
options = {
|
||||
booleans: true,
|
||||
|
||||
@@ -486,7 +486,7 @@ do_continue: {
|
||||
|
||||
catch_ie8_1: {
|
||||
options = {
|
||||
ie8: true,
|
||||
ie: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
@@ -506,7 +506,7 @@ catch_ie8_1: {
|
||||
catch_ie8_2: {
|
||||
options = {
|
||||
dead_code: true,
|
||||
ie8: true,
|
||||
ie: true,
|
||||
passes: 2,
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
@@ -727,7 +727,7 @@ issue_4193: {
|
||||
|
||||
issue_4195: {
|
||||
mangle = {
|
||||
ie8: true,
|
||||
ie: true,
|
||||
}
|
||||
input: {
|
||||
console.log(function f(a) {
|
||||
@@ -1131,7 +1131,7 @@ issue_4225: {
|
||||
|
||||
issue_4229: {
|
||||
options = {
|
||||
ie8: true,
|
||||
ie: true,
|
||||
side_effects: true,
|
||||
}
|
||||
input: {
|
||||
@@ -1151,7 +1151,7 @@ issue_4229: {
|
||||
|
||||
issue_4231: {
|
||||
options = {
|
||||
ie8: true,
|
||||
ie: true,
|
||||
side_effects: true,
|
||||
}
|
||||
input: {
|
||||
@@ -1498,3 +1498,197 @@ issue_4691: {
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
issue_4848: {
|
||||
options = {
|
||||
if_return: true,
|
||||
}
|
||||
input: {
|
||||
function f(a) {
|
||||
a(function() {
|
||||
console.log(b);
|
||||
});
|
||||
if (!console)
|
||||
return;
|
||||
const b = "PASS";
|
||||
}
|
||||
var g;
|
||||
f(function(h) {
|
||||
g = h;
|
||||
});
|
||||
g();
|
||||
}
|
||||
expect: {
|
||||
function f(a) {
|
||||
a(function() {
|
||||
console.log(b);
|
||||
});
|
||||
if (!console)
|
||||
return;
|
||||
const b = "PASS";
|
||||
}
|
||||
var g;
|
||||
f(function(h) {
|
||||
g = h;
|
||||
});
|
||||
g();
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
issue_4954_1: {
|
||||
rename = true
|
||||
input: {
|
||||
"use strict";
|
||||
(function() {
|
||||
{
|
||||
const a = "foo";
|
||||
console.log(a);
|
||||
}
|
||||
{
|
||||
const a = "bar";
|
||||
console.log(a);
|
||||
}
|
||||
})();
|
||||
}
|
||||
expect: {
|
||||
"use strict";
|
||||
(function() {
|
||||
{
|
||||
const a = "foo";
|
||||
console.log(a);
|
||||
}
|
||||
{
|
||||
const b = "bar";
|
||||
console.log(b);
|
||||
}
|
||||
})();
|
||||
}
|
||||
expect_stdout: [
|
||||
"foo",
|
||||
"bar",
|
||||
]
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
issue_4954_2: {
|
||||
mangle = {}
|
||||
input: {
|
||||
"use strict";
|
||||
const a = null;
|
||||
(function(b) {
|
||||
for (const a in null);
|
||||
for (const a in b)
|
||||
console.log("PASS");
|
||||
})([ null ]);
|
||||
}
|
||||
expect: {
|
||||
"use strict";
|
||||
const a = null;
|
||||
(function(o) {
|
||||
for (const n in null);
|
||||
for (const n in o)
|
||||
console.log("PASS");
|
||||
})([ null ]);
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
issue_4960: {
|
||||
mangle = {}
|
||||
input: {
|
||||
"use strict";
|
||||
var a;
|
||||
(function() {
|
||||
{
|
||||
const a = console.log("PASS");
|
||||
}
|
||||
try {} catch (e) {
|
||||
const a = console.log("FAIL");
|
||||
}
|
||||
})();
|
||||
}
|
||||
expect: {
|
||||
"use strict";
|
||||
var a;
|
||||
(function() {
|
||||
{
|
||||
const o = console.log("PASS");
|
||||
}
|
||||
try {} catch (o) {
|
||||
const c = console.log("FAIL");
|
||||
}
|
||||
})();
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
issue_4965_1: {
|
||||
mangle = {}
|
||||
input: {
|
||||
"use strict";
|
||||
try {
|
||||
c;
|
||||
} catch (a) {
|
||||
{
|
||||
const a = 1;
|
||||
}
|
||||
{
|
||||
const a = console.log(typeof c);
|
||||
}
|
||||
}
|
||||
}
|
||||
expect: {
|
||||
"use strict";
|
||||
try {
|
||||
c;
|
||||
} catch (t) {
|
||||
{
|
||||
const c = 1;
|
||||
}
|
||||
{
|
||||
const t = console.log(typeof c);
|
||||
}
|
||||
}
|
||||
}
|
||||
expect_stdout: "undefined"
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
issue_4965_2: {
|
||||
mangle = {}
|
||||
input: {
|
||||
"use strict";
|
||||
try {
|
||||
throw 1;
|
||||
} catch (e) {
|
||||
try {
|
||||
{
|
||||
const e = 2;
|
||||
}
|
||||
} finally {
|
||||
const e = 3;
|
||||
console.log(typeof t);
|
||||
}
|
||||
}
|
||||
}
|
||||
expect: {
|
||||
"use strict";
|
||||
try {
|
||||
throw 1;
|
||||
} catch (o) {
|
||||
try {
|
||||
{
|
||||
const t = 2;
|
||||
}
|
||||
} finally {
|
||||
const o = 3;
|
||||
console.log(typeof t);
|
||||
}
|
||||
}
|
||||
}
|
||||
expect_stdout: "undefined"
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
@@ -939,6 +939,185 @@ catch_return_assign: {
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
catch_return_assign_may_throw: {
|
||||
options = {
|
||||
dead_code: true,
|
||||
}
|
||||
input: {
|
||||
function f() {
|
||||
try {
|
||||
throw "FAIL";
|
||||
} catch (e) {
|
||||
return e = console.log("PASS");
|
||||
}
|
||||
}
|
||||
f();
|
||||
}
|
||||
expect: {
|
||||
function f() {
|
||||
try {
|
||||
throw "FAIL";
|
||||
} catch (e) {
|
||||
return console.log("PASS");
|
||||
}
|
||||
}
|
||||
f();
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
finally_return_assign: {
|
||||
options = {
|
||||
dead_code: true,
|
||||
}
|
||||
input: {
|
||||
console.log(function(a) {
|
||||
try {
|
||||
throw "FAIL";
|
||||
} finally {
|
||||
return a = "PASS";
|
||||
}
|
||||
}());
|
||||
}
|
||||
expect: {
|
||||
console.log(function(a) {
|
||||
try {
|
||||
throw "FAIL";
|
||||
} finally {
|
||||
return "PASS";
|
||||
}
|
||||
}());
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
last_assign_statement: {
|
||||
options = {
|
||||
dead_code: true,
|
||||
}
|
||||
input: {
|
||||
function f(a) {
|
||||
a = a("PASS");
|
||||
}
|
||||
f(console.log);
|
||||
}
|
||||
expect: {
|
||||
function f(a) {
|
||||
a("PASS");
|
||||
}
|
||||
f(console.log);
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
last_assign_if_else: {
|
||||
options = {
|
||||
dead_code: true,
|
||||
}
|
||||
input: {
|
||||
function f(a) {
|
||||
if (a)
|
||||
a = console.log("foo");
|
||||
else {
|
||||
console.log("bar");
|
||||
a = console.log("baz");
|
||||
}
|
||||
}
|
||||
f(42);
|
||||
f(null);
|
||||
}
|
||||
expect: {
|
||||
function f(a) {
|
||||
if (a)
|
||||
console.log("foo");
|
||||
else {
|
||||
console.log("bar");
|
||||
console.log("baz");
|
||||
}
|
||||
}
|
||||
f(42);
|
||||
f(null);
|
||||
}
|
||||
expect_stdout: [
|
||||
"foo",
|
||||
"bar",
|
||||
"baz",
|
||||
]
|
||||
}
|
||||
|
||||
last_assign_catch: {
|
||||
options = {
|
||||
dead_code: true,
|
||||
}
|
||||
input: {
|
||||
function f() {
|
||||
try {
|
||||
throw "FAIL";
|
||||
} catch (e) {
|
||||
e = console.log("PASS");
|
||||
}
|
||||
}
|
||||
f();
|
||||
}
|
||||
expect: {
|
||||
function f() {
|
||||
try {
|
||||
throw "FAIL";
|
||||
} catch (e) {
|
||||
console.log("PASS");
|
||||
}
|
||||
}
|
||||
f();
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
last_assign_finally: {
|
||||
options = {
|
||||
dead_code: true,
|
||||
}
|
||||
input: {
|
||||
function f(a) {
|
||||
try {
|
||||
throw a.log;
|
||||
} catch (e) {
|
||||
a = e;
|
||||
} finally {
|
||||
a = a("PASS");
|
||||
}
|
||||
}
|
||||
f(console);
|
||||
}
|
||||
expect: {
|
||||
function f(a) {
|
||||
try {
|
||||
throw a.log;
|
||||
} catch (e) {
|
||||
a = e;
|
||||
} finally {
|
||||
a("PASS");
|
||||
}
|
||||
}
|
||||
f(console);
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
consecutive_assignments: {
|
||||
options = {
|
||||
dead_code: true,
|
||||
}
|
||||
input: {
|
||||
while (a = void 0, a = "PASS", console.log(a));
|
||||
var a;
|
||||
}
|
||||
expect: {
|
||||
while (void 0, a = "PASS", console.log(a));
|
||||
var a;
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
issue_3578: {
|
||||
options = {
|
||||
dead_code: true,
|
||||
@@ -1420,3 +1599,73 @@ issue_4570: {
|
||||
}
|
||||
expect_stdout: "NaN"
|
||||
}
|
||||
|
||||
issue_5030: {
|
||||
options = {
|
||||
dead_code: true,
|
||||
}
|
||||
input: {
|
||||
(function(a, b) {
|
||||
a = function f() {
|
||||
if (a)
|
||||
if (b--)
|
||||
setImmediate(f);
|
||||
else
|
||||
console.log("FAIL");
|
||||
else
|
||||
console.log("PASS");
|
||||
}();
|
||||
})(42, 1);
|
||||
}
|
||||
expect: {
|
||||
(function(a, b) {
|
||||
a = function f() {
|
||||
if (a)
|
||||
if (b--)
|
||||
setImmediate(f);
|
||||
else
|
||||
console.log("FAIL");
|
||||
else
|
||||
console.log("PASS");
|
||||
}();
|
||||
})(42, 1);
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=0.12"
|
||||
}
|
||||
|
||||
issue_5106_1: {
|
||||
options = {
|
||||
dead_code: true,
|
||||
}
|
||||
input: {
|
||||
console.log(typeof function(a) {
|
||||
return a = arguments;
|
||||
}("FAIL")[0]);
|
||||
}
|
||||
expect: {
|
||||
console.log(typeof function(a) {
|
||||
return a = arguments;
|
||||
}("FAIL")[0]);
|
||||
}
|
||||
expect_stdout: "object"
|
||||
}
|
||||
|
||||
issue_5106_2: {
|
||||
options = {
|
||||
dead_code: true,
|
||||
}
|
||||
input: {
|
||||
"use strict";
|
||||
console.log(function(a) {
|
||||
return a = arguments;
|
||||
}("PASS")[0]);
|
||||
}
|
||||
expect: {
|
||||
"use strict";
|
||||
console.log(function(a) {
|
||||
return arguments;
|
||||
}("PASS")[0]);
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
@@ -70,7 +70,7 @@ object_shorthand_assign: {
|
||||
({ a = "PASS" } = 42);
|
||||
console.log(a);
|
||||
}
|
||||
expect_exact: '({a:a="PASS"}=42);console.log(a);'
|
||||
expect_exact: '({a="PASS"}=42);console.log(a);'
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=6"
|
||||
}
|
||||
@@ -80,7 +80,7 @@ object_shorthand_declaration: {
|
||||
var { a = "PASS" } = 42;
|
||||
console.log(a);
|
||||
}
|
||||
expect_exact: 'var{a:a="PASS"}=42;console.log(a);'
|
||||
expect_exact: 'var{a="PASS"}=42;console.log(a);'
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=6"
|
||||
}
|
||||
@@ -91,7 +91,7 @@ object_shorthand_function: {
|
||||
console.log(a);
|
||||
})(42);
|
||||
}
|
||||
expect_exact: '(function({a:a="PASS"}){console.log(a)})(42);'
|
||||
expect_exact: '(function({a="PASS"}){console.log(a)})(42);'
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=6"
|
||||
}
|
||||
@@ -149,7 +149,7 @@ process_boolean_returns: {
|
||||
}
|
||||
expect: {
|
||||
console.log(function(a = console.log("FAIL 1")) {
|
||||
return a() ? "PASS" : "FAIL 2";
|
||||
return 42 ? "PASS" : "FAIL 2";
|
||||
}(function() {
|
||||
return 1;
|
||||
}));
|
||||
@@ -245,21 +245,64 @@ maintain_if: {
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
reduce_value: {
|
||||
reduce_funarg: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
keep_fargs: false,
|
||||
reduce_vars: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
console.log(function(a = "PASS") {
|
||||
return a;
|
||||
}());
|
||||
console.log(...function(a = "foo", b = "bar", c = "baz") {
|
||||
return [ a, b, c ];
|
||||
}(void 0, null));
|
||||
}
|
||||
expect: {
|
||||
console.log("PASS");
|
||||
console.log(...function() {
|
||||
return [ "foo", null, "baz" ];
|
||||
}());
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
expect_stdout: "foo null baz"
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
reduce_array: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
reduce_vars: true,
|
||||
toplevel: true,
|
||||
unsafe: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var [ a = "foo", b = "bar", c = "baz" ] = [ void 0, null ];
|
||||
console.log(a, b, c);
|
||||
}
|
||||
expect: {
|
||||
var [ c = "baz" ] = [];
|
||||
console.log("foo", null, c);
|
||||
}
|
||||
expect_stdout: "foo null baz"
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
reduce_object: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
reduce_vars: true,
|
||||
toplevel: true,
|
||||
unsafe: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var { a = "foo", b = "bar", c = "baz" } = { a: void 0, b: null };
|
||||
console.log(a, b, c);
|
||||
}
|
||||
expect: {
|
||||
var { c = "baz" } = {};
|
||||
console.log("foo", null, c);
|
||||
}
|
||||
expect_stdout: "foo null baz"
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
@@ -422,7 +465,9 @@ inline_loop_1: {
|
||||
inline_loop_2: {
|
||||
options = {
|
||||
inline: true,
|
||||
sequences: true,
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
while (function(a = [ "PASS" ]) {
|
||||
@@ -432,10 +477,11 @@ inline_loop_2: {
|
||||
}());
|
||||
}
|
||||
expect: {
|
||||
while (a = [ "PASS" ], a = function f(b) {
|
||||
console.log(a[b]);
|
||||
}(0), void 0) ;
|
||||
var a;
|
||||
while (a = [ "PASS" ],
|
||||
b = void 0,
|
||||
b = 0,
|
||||
void (a = void console.log(a[b])));
|
||||
var a, b;
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=6"
|
||||
@@ -522,6 +568,20 @@ retain_empty_iife: {
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
drop_new_function: {
|
||||
options = {
|
||||
side_effects: true,
|
||||
}
|
||||
input: {
|
||||
new function(a = console.log("PASS")) {}();
|
||||
}
|
||||
expect: {
|
||||
void console.log("PASS");
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
retain_fargs: {
|
||||
options = {
|
||||
unused: true,
|
||||
@@ -598,6 +658,7 @@ unused_var_1: {
|
||||
|
||||
unused_var_2: {
|
||||
options = {
|
||||
pure_getters: "strict",
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
}
|
||||
@@ -609,11 +670,7 @@ unused_var_2: {
|
||||
};
|
||||
}
|
||||
expect: {
|
||||
var {
|
||||
p: [] = [ console.log("FAIL") ],
|
||||
} = {
|
||||
p: [ console.log("PASS") ],
|
||||
};
|
||||
console.log("PASS");
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=6"
|
||||
@@ -676,7 +733,7 @@ unused_value_var_2: {
|
||||
console.log(a);
|
||||
}
|
||||
expect: {
|
||||
var [ a ] = [ "PASS" ];
|
||||
var a = [ "PASS" ][0];
|
||||
console.log(a);
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
@@ -1257,7 +1314,7 @@ issue_4468: {
|
||||
expect: {
|
||||
(function() {
|
||||
var {
|
||||
[console.log("PASS")]: b = 0,
|
||||
[console.log("PASS")]: b,
|
||||
} = 0;
|
||||
})();
|
||||
}
|
||||
@@ -1416,6 +1473,7 @@ issue_4502_1: {
|
||||
expect: {
|
||||
(function() {
|
||||
var a = "PASS";
|
||||
void 0,
|
||||
console.log(a),
|
||||
a++,
|
||||
void 0;
|
||||
@@ -1439,6 +1497,7 @@ issue_4502_2: {
|
||||
expect: {
|
||||
(function() {
|
||||
var a = "PASS";
|
||||
void 0,
|
||||
console.log(a),
|
||||
a++,
|
||||
void 0;
|
||||
@@ -1661,3 +1720,187 @@ issue_4588_2_evaluate: {
|
||||
expect_stdout: "1"
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
issue_4817: {
|
||||
options = {
|
||||
ie: true,
|
||||
inline: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
(function f(a = console.log(typeof f)) {
|
||||
return 42;
|
||||
})();
|
||||
}
|
||||
expect: {
|
||||
(function f(a = console.log(typeof f)) {
|
||||
return 42;
|
||||
})();
|
||||
}
|
||||
expect_stdout: "function"
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
issue_4854: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
inline: true,
|
||||
side_effects: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
console.log(function(a) {
|
||||
(function(b = a = "foo") {
|
||||
[] = "foo";
|
||||
})();
|
||||
a;
|
||||
}());
|
||||
}
|
||||
expect: {
|
||||
console.log(void 0);
|
||||
}
|
||||
expect_stdout: "undefined"
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
issue_4916: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
pure_getters: "strict",
|
||||
reduce_vars: true,
|
||||
}
|
||||
input: {
|
||||
var log = console.log;
|
||||
(function(b = "foo") {
|
||||
b.value = "FAIL";
|
||||
b;
|
||||
log(b.value);
|
||||
})();
|
||||
}
|
||||
expect: {
|
||||
var log = console.log;
|
||||
(function(b = "foo") {
|
||||
b.value = "FAIL";
|
||||
b;
|
||||
log(b.value);
|
||||
})();
|
||||
}
|
||||
expect_stdout: "undefined"
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
issue_4994: {
|
||||
options = {
|
||||
loops: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var a = "FAIL";
|
||||
(function(b = function() {
|
||||
for (a in { PASS: 42 });
|
||||
}()) {
|
||||
var a;
|
||||
})();
|
||||
console.log(a);
|
||||
}
|
||||
expect: {
|
||||
var a = "FAIL";
|
||||
(function(b = function() {
|
||||
for (a in { PASS: 42 });
|
||||
}()) {})();
|
||||
console.log(a);
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
issue_5057_1: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
inline: true,
|
||||
sequences: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var a = 42;
|
||||
(function() {
|
||||
var b = function(c = (console.log("foo"), b = a)) {
|
||||
a && console.log("bar");
|
||||
}();
|
||||
})();
|
||||
}
|
||||
expect: {
|
||||
var a = 42;
|
||||
console.log("foo"),
|
||||
void (a && console.log("bar"));
|
||||
}
|
||||
expect_stdout: [
|
||||
"foo",
|
||||
"bar",
|
||||
]
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
issue_5057_2: {
|
||||
options = {
|
||||
inline: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
(function f(a) {
|
||||
(function(b = console.log("FAIL")) {})(a);
|
||||
})(42);
|
||||
console.log(typeof b);
|
||||
}
|
||||
expect: {
|
||||
(function(a) {
|
||||
[ b = console.log("FAIL") ] = [ a ],
|
||||
void 0;
|
||||
var b;
|
||||
})(42);
|
||||
console.log(typeof b);
|
||||
}
|
||||
expect_stdout: "undefined"
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
issue_5057_3: {
|
||||
options = {
|
||||
inline: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
(function(a) {
|
||||
(function f(b) {
|
||||
(function(a = console.log("FAIL 1")) {})(b);
|
||||
console.log(a);
|
||||
})("FAIL 2");
|
||||
})("PASS");
|
||||
}
|
||||
expect: {
|
||||
(function(a) {
|
||||
(function(b) {
|
||||
(function(a = console.log("FAIL 1")) {})(b);
|
||||
console.log(a);
|
||||
})("FAIL 2");
|
||||
})("PASS");
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
issue_5065: {
|
||||
options = {
|
||||
pure_getters: "strict",
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var [ a = console.log("PASS") ] = [ (A = 42).p ];
|
||||
}
|
||||
expect: {
|
||||
var [ a = console.log("PASS") ] = [ (A = 42).p ];
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
@@ -1010,6 +1010,42 @@ collapse_vars_8: {
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
collapse_vars_9: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
}
|
||||
input: {
|
||||
console.log(function(a) {
|
||||
try {
|
||||
var b = function([ c ]) {
|
||||
if (c)
|
||||
return "FAIL 1";
|
||||
}();
|
||||
a = "FAIL 2";
|
||||
return b;
|
||||
} catch (e) {
|
||||
return a;
|
||||
}
|
||||
}("PASS"));
|
||||
}
|
||||
expect: {
|
||||
console.log(function(a) {
|
||||
try {
|
||||
var b = function([ c ]) {
|
||||
if (c)
|
||||
return "FAIL 1";
|
||||
}();
|
||||
a = "FAIL 2";
|
||||
return b;
|
||||
} catch (e) {
|
||||
return a;
|
||||
}
|
||||
}("PASS"));
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
conditionals: {
|
||||
options = {
|
||||
conditionals: true,
|
||||
@@ -1069,7 +1105,7 @@ drop_unused_1: {
|
||||
try {
|
||||
throw 42;
|
||||
} catch (a) {
|
||||
var [ a ] = [];
|
||||
var a = [][0];
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -1102,6 +1138,175 @@ drop_unused_2: {
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
drop_hole: {
|
||||
options = {
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var [ a ] = [ , ];
|
||||
console.log(a);
|
||||
}
|
||||
expect: {
|
||||
var a = [][0];
|
||||
console.log(a);
|
||||
}
|
||||
expect_stdout: "undefined"
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
keep_key_1: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
side_effects: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
({} = {
|
||||
[(console.log("PASS"), 42)]: null,
|
||||
});
|
||||
}
|
||||
expect: {
|
||||
({} = {
|
||||
[(console.log("PASS"), 42)]: 0,
|
||||
});
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
keep_key_2: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var { 42: a } = { [(console.log("PASS"), 42)](){} };
|
||||
}
|
||||
expect: {
|
||||
var {} = { [(console.log("PASS"), 42)]: 0 };
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
keep_key_2_pure_getters: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
pure_getters: "strict",
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var { 42: a } = { [(console.log("PASS"), 42)](){} };
|
||||
}
|
||||
expect: {
|
||||
console.log("PASS");
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
keep_reference: {
|
||||
options = {
|
||||
reduce_vars: true,
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var a = [ {}, 42 ];
|
||||
var [ b, c ] = a;
|
||||
console.log(a[0] === b ? "PASS" : "FAIL");
|
||||
}
|
||||
expect: {
|
||||
var a = [ {}, 42 ];
|
||||
var [ b ] = a;
|
||||
console.log(a[0] === b ? "PASS" : "FAIL");
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
maintain_position_assign: {
|
||||
options = {
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
console.log(([ , ] = [ , "PASS" ])[1]);
|
||||
}
|
||||
expect: {
|
||||
console.log([ , "PASS" ][1]);
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
maintain_position_var: {
|
||||
options = {
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
A = "FAIL";
|
||||
var [ a, b ] = [ A ];
|
||||
console.log(b || "PASS");
|
||||
}
|
||||
expect: {
|
||||
A = "FAIL";
|
||||
var [ , b ] = [ A ];
|
||||
console.log(b || "PASS");
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
side_effects_array: {
|
||||
options = {
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
try {
|
||||
var [ a ] = 42;
|
||||
} catch (e) {
|
||||
console.log("PASS");
|
||||
}
|
||||
}
|
||||
expect: {
|
||||
try {
|
||||
var [ a ] = 42;
|
||||
} catch (e) {
|
||||
console.log("PASS");
|
||||
}
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
side_effects_object: {
|
||||
options = {
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var a = null, b = console, { c } = 42;
|
||||
try {
|
||||
c[a = "PASS"];
|
||||
} catch (e) {
|
||||
console.log(a);
|
||||
}
|
||||
}
|
||||
expect: {
|
||||
var a = null, c = (console, 42["c"]);
|
||||
try {
|
||||
c[a = "PASS"];
|
||||
} catch (e) {
|
||||
console.log(a);
|
||||
}
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
join_vars: {
|
||||
options = {
|
||||
conditionals: true,
|
||||
@@ -1824,7 +2029,7 @@ issue_4321: {
|
||||
|
||||
issue_4323: {
|
||||
options = {
|
||||
ie8: true,
|
||||
ie: true,
|
||||
inline: true,
|
||||
merge_vars: true,
|
||||
reduce_vars: true,
|
||||
@@ -2556,3 +2761,367 @@ issue_4608_2: {
|
||||
expect_stdout: "f"
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
issue_4994: {
|
||||
options = {
|
||||
loops: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var a = "FAIL";
|
||||
(function([
|
||||
{
|
||||
[function() {
|
||||
for (a in { PASS: null });
|
||||
}()]: b,
|
||||
},
|
||||
]) {
|
||||
var a;
|
||||
})([ 42 ]);
|
||||
console.log(a);
|
||||
}
|
||||
expect: {
|
||||
var a = "FAIL";
|
||||
(function([
|
||||
{
|
||||
[function() {
|
||||
for (a in { PASS: null });
|
||||
}()]: b,
|
||||
},
|
||||
]) {})([ 42 ]);
|
||||
console.log(a);
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
issue_5017: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
reduce_vars: true,
|
||||
toplevel: true,
|
||||
}
|
||||
input: {
|
||||
var a = function() {};
|
||||
var b = c = a;
|
||||
var c = [ c ] = [ c ];
|
||||
console.log(c[0] === a ? "PASS" : "FAIL");
|
||||
}
|
||||
expect: {
|
||||
var a = function() {};
|
||||
var b = a;
|
||||
var c = [ c ] = [ c = a ];
|
||||
console.log(c[0] === a ? "PASS" : "FAIL");
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
issue_5071_1: {
|
||||
options = {
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var a;
|
||||
console.log(([ , a ] = [ "PA", , ]).join("SS"));
|
||||
}
|
||||
expect: {
|
||||
var a;
|
||||
console.log(([ , a ] = [ "PA", , ]).join("SS"));
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
issue_5071_2: {
|
||||
options = {
|
||||
pure_getters: "strict",
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var a;
|
||||
([ a ] = []).p = console.log("PASS");
|
||||
}
|
||||
expect: {
|
||||
var a;
|
||||
([ a ] = []).p = console.log("PASS");
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
issue_5074_getter: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
side_effects: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
({} = { get [(console.log("PASS"), 42)]() {} });
|
||||
}
|
||||
expect: {
|
||||
({} = { [(console.log("PASS"), 42)]: 0 });
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
issue_5074_getter_pure_getters: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
pure_getters: "strict",
|
||||
side_effects: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
({} = { get [(console.log("PASS"), 42)]() {} });
|
||||
}
|
||||
expect: {
|
||||
console.log("PASS");
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
issue_5074_setter: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
side_effects: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
({} = { set [(console.log("PASS"), 42)](v) {} });
|
||||
}
|
||||
expect: {
|
||||
({} = { [(console.log("PASS"), 42)]: 0 });
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
issue_5074_setter_pure_getters: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
pure_getters: "strict",
|
||||
side_effects: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
({} = { set [(console.log("PASS"), 42)](v) {} });
|
||||
}
|
||||
expect: {
|
||||
console.log("PASS");
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
issue_5074_method: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
side_effects: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
({} = { [(console.log("PASS"), 42)]() {} });
|
||||
}
|
||||
expect: {
|
||||
({} = { [(console.log("PASS"), 42)]: 0 });
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
issue_5074_method_pure_getters: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
pure_getters: "strict",
|
||||
side_effects: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
({} = { [(console.log("PASS"), 42)]() {} });
|
||||
}
|
||||
expect: {
|
||||
console.log("PASS");
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
issue_5085_1: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
reduce_vars: true,
|
||||
toplevel: true,
|
||||
unsafe: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var a = "PASS";
|
||||
var [ b ] = [ 42, a ], c = b ? 0 : a = "FAIL";
|
||||
console.log(a);
|
||||
}
|
||||
expect: {
|
||||
var a = "PASS";
|
||||
var b = [ 42 ][0];
|
||||
b;
|
||||
console.log(a);
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
issue_5085_2: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
reduce_vars: true,
|
||||
side_effects: true,
|
||||
unsafe: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var a = "PASS";
|
||||
(function(b) {
|
||||
[ b ] = [ 42, a ];
|
||||
var c = b ? 0 : a = "FAIL";
|
||||
})();
|
||||
console.log(a);
|
||||
}
|
||||
expect: {
|
||||
var a = "PASS";
|
||||
(function(b) {
|
||||
b = [ 42 ][0];
|
||||
})();
|
||||
console.log(a);
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
issue_5087_1: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
inline: true,
|
||||
properties: true,
|
||||
sequences: true,
|
||||
side_effects: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var a = "PASS";
|
||||
(function() {
|
||||
(function() {
|
||||
(function([ b ]) {
|
||||
b && console.log(b);
|
||||
})([ a ]);
|
||||
})();
|
||||
})();
|
||||
}
|
||||
expect: {
|
||||
var a = "PASS";
|
||||
(function() {
|
||||
var b;
|
||||
(b = a) && console.log(b);
|
||||
})();
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
issue_5087_2: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
inline: true,
|
||||
passes: 2,
|
||||
properties: true,
|
||||
reduce_vars: true,
|
||||
sequences: true,
|
||||
side_effects: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var a = "PASS";
|
||||
(function() {
|
||||
(function() {
|
||||
(function([ b ]) {
|
||||
b && console.log(b);
|
||||
})([ a ]);
|
||||
})();
|
||||
})();
|
||||
}
|
||||
expect: {
|
||||
var a = "PASS";
|
||||
a && console.log(a);
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
issue_5114_1: {
|
||||
options = {
|
||||
inline: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var a = "PASS";
|
||||
(function({}, a) {})(42);
|
||||
console.log(a);
|
||||
}
|
||||
expect: {
|
||||
var a = "PASS";
|
||||
[ {} ] = [ 42 ],
|
||||
void 0;
|
||||
console.log(a);
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
issue_5114_2: {
|
||||
options = {
|
||||
inline: true,
|
||||
pure_getters: "strict",
|
||||
reduce_vars: true,
|
||||
side_effects: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var a = "PASS";
|
||||
(function f([], a) {
|
||||
f.length;
|
||||
})([]);
|
||||
console.log(a);
|
||||
}
|
||||
expect: {
|
||||
var a = "PASS";
|
||||
0;
|
||||
console.log(a);
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
issue_5114_3: {
|
||||
options = {
|
||||
inline: true,
|
||||
pure_getters: "strict",
|
||||
reduce_vars: true,
|
||||
side_effects: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var a = "PASS";
|
||||
(function f(a, {}) {
|
||||
f.length;
|
||||
})(null, 42);
|
||||
console.log(a);
|
||||
}
|
||||
expect: {
|
||||
var a = "PASS";
|
||||
0;
|
||||
console.log(a);
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
@@ -795,9 +795,9 @@ issue_1656: {
|
||||
beautify: true,
|
||||
}
|
||||
input: {
|
||||
for(var a=0;;);
|
||||
for (var a=0;;);
|
||||
}
|
||||
expect_exact: "for (;;) ;"
|
||||
expect_exact: "for (;;);"
|
||||
}
|
||||
|
||||
issue_1709: {
|
||||
@@ -1110,7 +1110,7 @@ issue_1838: {
|
||||
}
|
||||
expect_exact: [
|
||||
"function f() {",
|
||||
" for (a; c; ) ;",
|
||||
" for (a; c; );",
|
||||
"}",
|
||||
]
|
||||
}
|
||||
@@ -1557,9 +1557,9 @@ issue_2665: {
|
||||
}
|
||||
expect: {
|
||||
var a = 1;
|
||||
!function g() {
|
||||
(function g() {
|
||||
a-- && g();
|
||||
}();
|
||||
})();
|
||||
console.log(a);
|
||||
}
|
||||
expect_stdout: "-1"
|
||||
@@ -2345,7 +2345,7 @@ function_argument_reference: {
|
||||
|
||||
function_parameter_ie8: {
|
||||
options = {
|
||||
ie8: true,
|
||||
ie: true,
|
||||
reduce_vars: true,
|
||||
unused: true,
|
||||
}
|
||||
@@ -2371,6 +2371,7 @@ function_parameter_ie8: {
|
||||
issue_3664: {
|
||||
options = {
|
||||
pure_getters: "strict",
|
||||
side_effects: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
@@ -2381,7 +2382,7 @@ issue_3664: {
|
||||
}
|
||||
expect: {
|
||||
console.log(function() {
|
||||
var b = ([ b && console.log("FAIL") ].p = 0, 0);
|
||||
var a, b = (a = (a = [ b && console.log("FAIL") ]).p = 0, 0);
|
||||
return "PASS";
|
||||
}());
|
||||
}
|
||||
@@ -2391,6 +2392,7 @@ issue_3664: {
|
||||
issue_3673: {
|
||||
options = {
|
||||
pure_getters: "strict",
|
||||
sequences: true,
|
||||
side_effects: true,
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
@@ -2401,8 +2403,6 @@ issue_3673: {
|
||||
console.log("PASS");
|
||||
}
|
||||
expect: {
|
||||
var a;
|
||||
(a = [ a ]).p = 42;
|
||||
console.log("PASS");
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
@@ -3080,11 +3080,10 @@ issue_4235: {
|
||||
})();
|
||||
}
|
||||
expect: {
|
||||
(function() {
|
||||
f = console.log(f),
|
||||
void 0;
|
||||
void function() {
|
||||
var f;
|
||||
})();
|
||||
console.log(f);
|
||||
}();
|
||||
}
|
||||
expect_stdout: "undefined"
|
||||
}
|
||||
@@ -3231,7 +3230,7 @@ issue_4558_1: {
|
||||
issue_4558_2: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
ie8: true,
|
||||
ie: true,
|
||||
reduce_vars: true,
|
||||
unused: true,
|
||||
}
|
||||
@@ -3274,3 +3273,222 @@ issue_4662: {
|
||||
}
|
||||
expect_stdout: "1 1"
|
||||
}
|
||||
|
||||
issue_4806_1: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
reduce_vars: true,
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
O = {
|
||||
f: function() {
|
||||
console.log(this === O ? "FAIL" : "PASS");
|
||||
},
|
||||
};
|
||||
var a;
|
||||
(a = 42, O.f)();
|
||||
a;
|
||||
}
|
||||
expect: {
|
||||
O = {
|
||||
f: function() {
|
||||
console.log(this === O ? "FAIL" : "PASS");
|
||||
},
|
||||
};
|
||||
(0, O.f)();
|
||||
42;
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
issue_4806_2: {
|
||||
options = {
|
||||
sequences: true,
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
O = {
|
||||
f: function() {
|
||||
console.log(this === O ? "FAIL" : "PASS");
|
||||
},
|
||||
};
|
||||
var a;
|
||||
(a = 42, O.f)();
|
||||
a;
|
||||
}
|
||||
expect: {
|
||||
O = {
|
||||
f: function() {
|
||||
console.log(this === O ? "FAIL" : "PASS");
|
||||
},
|
||||
},
|
||||
(0, O.f)();
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
issue_4806_3: {
|
||||
options = {
|
||||
side_effects: true,
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
O = {
|
||||
f: function() {
|
||||
console.log(this === O ? "FAIL" : "PASS");
|
||||
},
|
||||
};
|
||||
var a;
|
||||
(a = 42, O.f)();
|
||||
a;
|
||||
}
|
||||
expect: {
|
||||
O = {
|
||||
f: function() {
|
||||
console.log(this === O ? "FAIL" : "PASS");
|
||||
},
|
||||
};
|
||||
(0, O.f)();
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
issue_4834: {
|
||||
options = {
|
||||
inline: true,
|
||||
keep_fargs: false,
|
||||
pure_getters: "strict",
|
||||
reduce_vars: true,
|
||||
side_effects: true,
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
try {
|
||||
new function(a, b) {
|
||||
b;
|
||||
b.p;
|
||||
}(42);
|
||||
} catch (e) {
|
||||
console.log("PASS");
|
||||
}
|
||||
}
|
||||
expect: {
|
||||
try {
|
||||
b.p;
|
||||
} catch (e) {
|
||||
console.log("PASS");
|
||||
}
|
||||
var b;
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
issue_4912_1: {
|
||||
options = {
|
||||
pure_getters: "strict",
|
||||
reduce_vars: true,
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var a = A = function() {};
|
||||
A;
|
||||
a.prototype = {
|
||||
f: function() {
|
||||
console.log("PASS");
|
||||
},
|
||||
};
|
||||
new A().f();
|
||||
}
|
||||
expect: {
|
||||
var a = A = function() {};
|
||||
A;
|
||||
a.prototype = {
|
||||
f: function() {
|
||||
console.log("PASS");
|
||||
},
|
||||
};
|
||||
new A().f();
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
issue_4912_2: {
|
||||
options = {
|
||||
pure_getters: "strict",
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
console.log(function() {
|
||||
var g, f = function() {};
|
||||
f.p = {};
|
||||
(g = f.p.q = function() {}).r = "PASS";
|
||||
return f;
|
||||
}().p.q.r);
|
||||
}
|
||||
expect: {
|
||||
console.log(function() {
|
||||
var g, f = function() {};
|
||||
f.p = {};
|
||||
(f.p.q = function() {}).r = "PASS";
|
||||
return f;
|
||||
}().p.q.r);
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
issue_4912_3: {
|
||||
options = {
|
||||
pure_getters: "strict",
|
||||
reduce_vars: true,
|
||||
side_effects: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
console.log(function(f, g) {
|
||||
f = function() {};
|
||||
f.p = {};
|
||||
g = f.p.q = function() {};
|
||||
g.r = "PASS";
|
||||
return f;
|
||||
}().p.q.r);
|
||||
}
|
||||
expect: {
|
||||
console.log(function(f, g) {
|
||||
f = function() {};
|
||||
f.p = {};
|
||||
g = f.p.q = function() {};
|
||||
g.r = "PASS";
|
||||
return f;
|
||||
}().p.q.r);
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
issue_5079: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
pure_getters: "strict",
|
||||
reduce_vars: true,
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var a;
|
||||
do {
|
||||
(a = 123456).p = a;
|
||||
a.q = null;
|
||||
} while (console.log("PASS"));
|
||||
}
|
||||
expect: {
|
||||
do {
|
||||
0, 0, null;
|
||||
} while (console.log("PASS"));
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
@@ -684,26 +684,47 @@ prototype_function: {
|
||||
side_effects: true,
|
||||
}
|
||||
input: {
|
||||
var a = ({valueOf: 0}) < 1;
|
||||
var b = ({toString: 0}) < 1;
|
||||
var c = ({valueOf: 0}) + "";
|
||||
var d = ({toString: 0}) + "";
|
||||
var e = (({valueOf: 0}) + "")[2];
|
||||
var f = (({toString: 0}) + "")[2];
|
||||
var g = ({valueOf: 0}).valueOf();
|
||||
var h = ({toString: 0}).toString();
|
||||
function v() {
|
||||
return this.valueOf === v ? "PASS" : "FAIL";
|
||||
}
|
||||
console.log(({ valueOf: v }) < 1);
|
||||
console.log(({ valueOf: v }) + "");
|
||||
console.log((( {valueOf: v }) + "")[2]);
|
||||
console.log(({ valueOf: v }).valueOf());
|
||||
function t() {
|
||||
return this.toString === t ? "PASS" : "FAIL";
|
||||
}
|
||||
console.log(({ toString: t }) < 1);
|
||||
console.log(({ toString: t }) + "");
|
||||
console.log((( {toString: t }) + "")[2]);
|
||||
console.log(({ toString: t }).toString());
|
||||
}
|
||||
expect: {
|
||||
var a = ({valueOf: 0}) < 1;
|
||||
var b = ({toString: 0}) < 1;
|
||||
var c = ({valueOf: 0}) + "";
|
||||
var d = ({toString: 0}) + "";
|
||||
var e = (({valueOf: 0}) + "")[2];
|
||||
var f = (({toString: 0}) + "")[2];
|
||||
var g = 0();
|
||||
var h = 0();
|
||||
function v() {
|
||||
return this.valueOf === v ? "PASS" : "FAIL";
|
||||
}
|
||||
console.log(({ valueOf: v }) < 1);
|
||||
console.log(({ valueOf: v }) + "");
|
||||
console.log((( {valueOf: v }) + "")[2]);
|
||||
console.log(({ valueOf: v }).valueOf());
|
||||
function t() {
|
||||
return this.toString === t ? "PASS" : "FAIL";
|
||||
}
|
||||
console.log(({ toString: t }) < 1);
|
||||
console.log(({ toString: t }) + "");
|
||||
console.log((( {toString: t }) + "")[2]);
|
||||
console.log(({ toString: t }).toString());
|
||||
}
|
||||
expect_stdout: true
|
||||
expect_stdout: [
|
||||
"false",
|
||||
"PASS",
|
||||
"S",
|
||||
"PASS",
|
||||
"false",
|
||||
"PASS",
|
||||
"S",
|
||||
"PASS",
|
||||
]
|
||||
}
|
||||
|
||||
call_args: {
|
||||
@@ -3181,3 +3202,23 @@ issue_4552: {
|
||||
}
|
||||
expect_stdout: "NaN"
|
||||
}
|
||||
|
||||
issue_4886: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
unsafe: true,
|
||||
}
|
||||
input: {
|
||||
console.log("length" in {
|
||||
__proto__: function() {},
|
||||
length: void 0,
|
||||
});
|
||||
}
|
||||
expect: {
|
||||
console.log("length" in {
|
||||
__proto__: function() {},
|
||||
length: void 0,
|
||||
});
|
||||
}
|
||||
expect_stdout: "true"
|
||||
}
|
||||
|
||||
@@ -43,6 +43,28 @@ await: {
|
||||
node_version: ">=8"
|
||||
}
|
||||
|
||||
assignment_1: {
|
||||
input: {
|
||||
var a = 2;
|
||||
a **= 5;
|
||||
console.log(a);
|
||||
}
|
||||
expect_exact: "var a=2;a**=5;console.log(a);"
|
||||
expect_stdout: "32"
|
||||
node_version: ">=8"
|
||||
}
|
||||
|
||||
assignment_2: {
|
||||
input: {
|
||||
var a = 8n;
|
||||
a **= a;
|
||||
console.log(a);
|
||||
}
|
||||
expect_exact: "var a=8n;a**=a;console.log(a);"
|
||||
expect_stdout: "16777216n"
|
||||
node_version: ">=10"
|
||||
}
|
||||
|
||||
evaluate: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
@@ -77,8 +99,8 @@ issue_4664: {
|
||||
expect: {
|
||||
(function f() {
|
||||
new function(a) {
|
||||
console.log(typeof f, 2 ** 30, typeof this);
|
||||
}(0, A = 0);
|
||||
console.log(typeof f, a, typeof this);
|
||||
}((A = 0, 2 ** 30));
|
||||
})();
|
||||
}
|
||||
expect_stdout: "function 1073741824 object"
|
||||
|
||||
@@ -3,7 +3,7 @@ refs: {
|
||||
export {};
|
||||
export { a, b as B, c as case, d as default };
|
||||
}
|
||||
expect_exact: "export{};export{a as a,b as B,c as case,d as default};"
|
||||
expect_exact: "export{};export{a,b as B,c as case,d as default};"
|
||||
}
|
||||
|
||||
var_defs: {
|
||||
@@ -12,7 +12,7 @@ var_defs: {
|
||||
export let b = 2, c = 3;
|
||||
export var { d, e: [] } = f;
|
||||
}
|
||||
expect_exact: "export const a=1;export let b=2,c=3;export var{d:d,e:[]}=f;"
|
||||
expect_exact: "export const a=1;export let b=2,c=3;export var{d,e:[]}=f;"
|
||||
}
|
||||
|
||||
defuns: {
|
||||
@@ -35,7 +35,7 @@ defaults: {
|
||||
export default function*(a, b) {};
|
||||
export default async function f({ c }, ...[ d ]) {};
|
||||
}
|
||||
expect_exact: "export default 42;export default async;export default(x,y)=>x*x;export default class{}export default function*(a,b){}export default async function f({c:c},...[d]){}"
|
||||
expect_exact: "export default 42;export default async;export default(x,y)=>x*x;export default class{}export default function*(a,b){}export default async function f({c},...[d]){}"
|
||||
}
|
||||
|
||||
defaults_parentheses_1: {
|
||||
@@ -91,6 +91,13 @@ defaults_parentheses_6: {
|
||||
expect_exact: 'export default(function(){while(!console);})()?"FAIL":"PASS";'
|
||||
}
|
||||
|
||||
defaults_regexp: {
|
||||
input: {
|
||||
export default /foo/;
|
||||
}
|
||||
expect_exact: "export default/foo/;"
|
||||
}
|
||||
|
||||
foreign: {
|
||||
input: {
|
||||
export * from "foo";
|
||||
@@ -242,15 +249,15 @@ hoist_exports_2: {
|
||||
}
|
||||
}
|
||||
expect: {
|
||||
let f, { foo: o } = 42;
|
||||
function c(t, { [f]: a }) {
|
||||
t(a, c);
|
||||
let e, a = 42["foo"];
|
||||
function f(t, { [e]: o }) {
|
||||
t(o, f);
|
||||
}
|
||||
export default 42;
|
||||
export default async function e(t, ...{ [o]: a }) {
|
||||
(await t)(e, a);
|
||||
export default async function n(t, ...{ [a]: o }) {
|
||||
(await t)(n, o);
|
||||
};
|
||||
export { f as bbb, o as ccc, c as fff };
|
||||
export { e as bbb, a as ccc, f as fff };
|
||||
}
|
||||
}
|
||||
|
||||
@@ -399,6 +406,17 @@ single_use_class_default: {
|
||||
}
|
||||
}
|
||||
|
||||
hoist_funs: {
|
||||
options = {
|
||||
hoist_funs: true,
|
||||
}
|
||||
input: {
|
||||
export function f() {}
|
||||
export default async function* g() {}
|
||||
}
|
||||
expect_exact: "export function f(){}export default async function*g(){}"
|
||||
}
|
||||
|
||||
issue_4742_join_vars_1: {
|
||||
options = {
|
||||
join_vars: true,
|
||||
@@ -464,3 +482,17 @@ issue_4761: {
|
||||
}
|
||||
expect_exact: 'export default"function"==42;'
|
||||
}
|
||||
|
||||
issue_4766: {
|
||||
options = {
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var a = "foo";
|
||||
export var a = "bar";
|
||||
}
|
||||
expect: {
|
||||
var a = "foo";
|
||||
export var a = "bar";
|
||||
}
|
||||
}
|
||||
|
||||
@@ -1171,11 +1171,11 @@ issue_2620_4: {
|
||||
}
|
||||
expect: {
|
||||
var c = "FAIL";
|
||||
!function() {
|
||||
(function() {
|
||||
switch (NaN) {
|
||||
case void (c = "PASS"):
|
||||
}
|
||||
}();
|
||||
})();
|
||||
console.log(c);
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
@@ -2123,7 +2123,7 @@ issue_3016_2: {
|
||||
issue_3016_2_ie8: {
|
||||
options = {
|
||||
dead_code: true,
|
||||
ie8: true,
|
||||
ie: true,
|
||||
inline: true,
|
||||
toplevel: true,
|
||||
}
|
||||
@@ -2188,7 +2188,7 @@ issue_3016_3: {
|
||||
issue_3016_3_ie8: {
|
||||
options = {
|
||||
dead_code: true,
|
||||
ie8: true,
|
||||
ie: true,
|
||||
inline: true,
|
||||
toplevel: true,
|
||||
}
|
||||
@@ -2751,17 +2751,17 @@ functions: {
|
||||
function b() {
|
||||
return !!b;
|
||||
}
|
||||
var c = function(c) {
|
||||
function c(c) {
|
||||
return c;
|
||||
};
|
||||
}
|
||||
if (c(b(a()))) {
|
||||
function d() {}
|
||||
function e() {
|
||||
return typeof e;
|
||||
}
|
||||
var f = function(f) {
|
||||
function f(f) {
|
||||
return f;
|
||||
};
|
||||
}
|
||||
console.log(a(d()), b(e()), c(f(42)), typeof d, e(), typeof f);
|
||||
}
|
||||
}();
|
||||
@@ -2808,9 +2808,9 @@ functions_use_strict: {
|
||||
function b() {
|
||||
return !!b;
|
||||
}
|
||||
var c = function(c) {
|
||||
function c(c) {
|
||||
return c;
|
||||
};
|
||||
}
|
||||
if (c(b(a()))) {
|
||||
var d = function() {};
|
||||
var e = function y() {
|
||||
@@ -2826,6 +2826,60 @@ functions_use_strict: {
|
||||
expect_stdout: "a true 42 function function function"
|
||||
}
|
||||
|
||||
functions_cross_scope_reference: {
|
||||
options = {
|
||||
functions: true,
|
||||
reduce_vars: true,
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
log = function(fn) {
|
||||
console.log(typeof fn());
|
||||
};
|
||||
var a = function() {};
|
||||
function f() {
|
||||
return a;
|
||||
}
|
||||
while (log(f));
|
||||
}
|
||||
expect: {
|
||||
log = function(fn) {
|
||||
console.log(typeof fn());
|
||||
};
|
||||
function a() {}
|
||||
function f() {
|
||||
return a;
|
||||
}
|
||||
while (log(f));
|
||||
}
|
||||
expect_stdout: "function"
|
||||
}
|
||||
|
||||
functions_inner_var: {
|
||||
options = {
|
||||
functions: true,
|
||||
reduce_vars: true,
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var a = function() {
|
||||
var a;
|
||||
console.log(a, a);
|
||||
};
|
||||
a(a);
|
||||
}
|
||||
expect: {
|
||||
function a() {
|
||||
var a;
|
||||
console.log(a, a);
|
||||
}
|
||||
a();
|
||||
}
|
||||
expect_stdout: "undefined undefined"
|
||||
}
|
||||
|
||||
issue_2437: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
@@ -2881,6 +2935,7 @@ issue_2437: {
|
||||
issue_2485_1: {
|
||||
options = {
|
||||
functions: true,
|
||||
passes: 2,
|
||||
reduce_funcs: true,
|
||||
reduce_vars: true,
|
||||
unused: true,
|
||||
@@ -2931,6 +2986,7 @@ issue_2485_2: {
|
||||
options = {
|
||||
functions: true,
|
||||
inline: true,
|
||||
passes: 2,
|
||||
reduce_funcs: true,
|
||||
reduce_vars: true,
|
||||
unused: true,
|
||||
@@ -5212,7 +5268,7 @@ issue_4265: {
|
||||
expect: {
|
||||
function f() {
|
||||
return console, function() {
|
||||
return console.log(a);
|
||||
console.log(a);
|
||||
var a;
|
||||
}(), 0;
|
||||
}
|
||||
@@ -5753,22 +5809,120 @@ issue_4725_2: {
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
new_target: {
|
||||
new_target_1: {
|
||||
input: {
|
||||
console.log(typeof new function() {
|
||||
return new.target;
|
||||
}, function() {
|
||||
new function f() {
|
||||
console.log(new.target === f);
|
||||
}();
|
||||
console.log(function() {
|
||||
return new.target;
|
||||
}());
|
||||
}
|
||||
expect: {
|
||||
console.log(typeof new function() {
|
||||
return new.target;
|
||||
}(), function() {
|
||||
new function f() {
|
||||
console.log(new.target === f);
|
||||
}();
|
||||
console.log(function() {
|
||||
return new.target;
|
||||
}());
|
||||
}
|
||||
expect_stdout: "function undefined"
|
||||
expect_stdout: [
|
||||
"true",
|
||||
"undefined",
|
||||
]
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
new_target_2: {
|
||||
input: {
|
||||
new function(a) {
|
||||
if (!new.target)
|
||||
console.log("FAIL");
|
||||
else if (a)
|
||||
console.log("PASS");
|
||||
else
|
||||
new new.target(new.target.length);
|
||||
}();
|
||||
}
|
||||
expect: {
|
||||
new function(a) {
|
||||
if (!new.target)
|
||||
console.log("FAIL");
|
||||
else if (a)
|
||||
console.log("PASS");
|
||||
else
|
||||
new new.target(new.target.length);
|
||||
}();
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
new_target_collapse_vars: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
new function(a) {
|
||||
if (a)
|
||||
console.log("PASS");
|
||||
else
|
||||
new new.target(new.target.length);
|
||||
}(0);
|
||||
}
|
||||
expect: {
|
||||
new function(a) {
|
||||
if (a)
|
||||
console.log("PASS");
|
||||
else
|
||||
new new.target(new.target.length);
|
||||
}(0);
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
new_target_delete: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
}
|
||||
input: {
|
||||
new function() {
|
||||
console.log(delete new.target);
|
||||
}();
|
||||
}
|
||||
expect: {
|
||||
new function() {
|
||||
console.log(delete new.target);
|
||||
}();
|
||||
}
|
||||
expect_stdout: true
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
new_target_reduce_vars: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
reduce_vars: true,
|
||||
}
|
||||
input: {
|
||||
new function(a) {
|
||||
if (a)
|
||||
console.log("PASS");
|
||||
else
|
||||
new new.target(new.target.length);
|
||||
}(0);
|
||||
}
|
||||
expect: {
|
||||
new function(a) {
|
||||
if (a)
|
||||
console.log("PASS");
|
||||
else
|
||||
new new.target(new.target.length);
|
||||
}(0);
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
@@ -5832,3 +5986,646 @@ issue_4753_2: {
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
issue_4788: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
functions: true,
|
||||
keep_fnames: true,
|
||||
reduce_vars: true,
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
function f() {
|
||||
var a = function g() {
|
||||
if (0) {
|
||||
var g = 42;
|
||||
f();
|
||||
}
|
||||
g || console.log("PASS");
|
||||
};
|
||||
a(a);
|
||||
}
|
||||
f();
|
||||
}
|
||||
expect: {
|
||||
(function f() {
|
||||
function a() {
|
||||
if (0) {
|
||||
var g = 42;
|
||||
f();
|
||||
}
|
||||
g || console.log("PASS");
|
||||
}
|
||||
a();
|
||||
})();
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
issue_4823: {
|
||||
options = {
|
||||
functions: true,
|
||||
reduce_vars: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
console.log(typeof function() {
|
||||
{
|
||||
function f() {}
|
||||
var arguments = f();
|
||||
function g() {}
|
||||
var arguments = g;
|
||||
}
|
||||
return f && arguments;
|
||||
}());
|
||||
}
|
||||
expect: {
|
||||
console.log(typeof function() {
|
||||
{
|
||||
function f() {}
|
||||
f();
|
||||
var arguments = function() {};
|
||||
}
|
||||
return f && arguments;
|
||||
}());
|
||||
}
|
||||
expect_stdout: "function"
|
||||
}
|
||||
|
||||
drop_unused_self_reference: {
|
||||
options = {
|
||||
pure_getters: "strict",
|
||||
reduce_vars: true,
|
||||
sequences: true,
|
||||
side_effects: true,
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
function f() {}
|
||||
(f.p = f).q = console.log("PASS");
|
||||
}
|
||||
expect: {
|
||||
console.log("PASS");
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
reduce_cross_reference_1: {
|
||||
options = {
|
||||
passes: 3,
|
||||
pure_getters: "strict",
|
||||
reduce_vars: true,
|
||||
sequences: true,
|
||||
side_effects: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
(function(a, b) {
|
||||
a = b = function() {};
|
||||
a.p = a;
|
||||
b = a = function() {};
|
||||
b.q = b;
|
||||
})();
|
||||
}
|
||||
expect: {}
|
||||
expect_stdout: true
|
||||
}
|
||||
|
||||
reduce_cross_reference_1_toplevel: {
|
||||
options = {
|
||||
passes: 2,
|
||||
pure_getters: "strict",
|
||||
reduce_vars: true,
|
||||
sequences: true,
|
||||
side_effects: true,
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var a = b = function() {};
|
||||
a.p = a;
|
||||
var b = a = function() {};
|
||||
b.q = b;
|
||||
}
|
||||
expect: {}
|
||||
expect_stdout: true
|
||||
}
|
||||
|
||||
reduce_cross_reference_2: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
passes: 3,
|
||||
pure_getters: "strict",
|
||||
reduce_vars: true,
|
||||
sequences: true,
|
||||
side_effects: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
(function(a, b) {
|
||||
a = b = function() {};
|
||||
b.p = a;
|
||||
b = a = function() {};
|
||||
a.q = b;
|
||||
})();
|
||||
}
|
||||
expect: {}
|
||||
expect_stdout: true
|
||||
}
|
||||
|
||||
reduce_cross_reference_2_toplevel: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
passes: 2,
|
||||
pure_getters: "strict",
|
||||
reduce_vars: true,
|
||||
sequences: true,
|
||||
side_effects: true,
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var a = b = function() {};
|
||||
b.p = a;
|
||||
var b = a = function() {};
|
||||
a.q = b;
|
||||
}
|
||||
expect: {}
|
||||
expect_stdout: true
|
||||
}
|
||||
|
||||
reduce_cross_reference_3: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
passes: 3,
|
||||
pure_getters: "strict",
|
||||
reduce_vars: true,
|
||||
sequences: true,
|
||||
side_effects: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
(function(a, b) {
|
||||
a = b = function() {};
|
||||
a.p = b;
|
||||
b = a = function() {};
|
||||
b.q = a;
|
||||
})();
|
||||
}
|
||||
expect: {}
|
||||
expect_stdout: true
|
||||
}
|
||||
|
||||
reduce_cross_reference_3_toplevel: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
passes: 2,
|
||||
pure_getters: "strict",
|
||||
reduce_vars: true,
|
||||
sequences: true,
|
||||
side_effects: true,
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var a = b = function() {};
|
||||
a.p = b;
|
||||
var b = a = function() {};
|
||||
b.q = a;
|
||||
}
|
||||
expect: {}
|
||||
expect_stdout: true
|
||||
}
|
||||
|
||||
reduce_cross_reference_4: {
|
||||
options = {
|
||||
passes: 3,
|
||||
pure_getters: "strict",
|
||||
reduce_vars: true,
|
||||
sequences: true,
|
||||
side_effects: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
(function(a, b) {
|
||||
a = b = function() {};
|
||||
b.p = b;
|
||||
b = a = function() {};
|
||||
a.q = a;
|
||||
})();
|
||||
}
|
||||
expect: {}
|
||||
expect_stdout: true
|
||||
}
|
||||
|
||||
reduce_cross_reference_4_toplevel: {
|
||||
options = {
|
||||
passes: 2,
|
||||
pure_getters: "strict",
|
||||
reduce_vars: true,
|
||||
sequences: true,
|
||||
side_effects: true,
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var a = b = function() {};
|
||||
b.p = b;
|
||||
var b = a = function() {};
|
||||
a.q = a;
|
||||
}
|
||||
expect: {}
|
||||
expect_stdout: true
|
||||
}
|
||||
|
||||
recursive_collapse: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
reduce_vars: true,
|
||||
}
|
||||
input: {
|
||||
console.log(function f(a) {
|
||||
var b = a && f();
|
||||
return b;
|
||||
}("FAIL") || "PASS");
|
||||
}
|
||||
expect: {
|
||||
console.log(function f(a) {
|
||||
var b;
|
||||
return a && f();
|
||||
}("FAIL") || "PASS");
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
issue_5025: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
inline: true,
|
||||
reduce_vars: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
function f(a) {
|
||||
function g() {
|
||||
b = 42;
|
||||
}
|
||||
g(b = a);
|
||||
var b = this;
|
||||
console.log(typeof b);
|
||||
}
|
||||
f();
|
||||
}
|
||||
expect: {
|
||||
function f(a) {
|
||||
b = a,
|
||||
void (b = 42);
|
||||
var b = this;
|
||||
console.log(typeof b);
|
||||
}
|
||||
f();
|
||||
}
|
||||
expect_stdout: "object"
|
||||
}
|
||||
|
||||
issue_5036: {
|
||||
options = {
|
||||
functions: true,
|
||||
reduce_vars: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
console.log(typeof function() {
|
||||
var await = function f() {
|
||||
return f;
|
||||
};
|
||||
return await() === await;
|
||||
}() ? "PASS" : "FAIL");
|
||||
}
|
||||
expect: {
|
||||
console.log(typeof function() {
|
||||
function await() {
|
||||
return await;
|
||||
}
|
||||
return await() === await;
|
||||
}() ? "PASS" : "FAIL");
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
issue_5046: {
|
||||
options = {
|
||||
conditionals: true,
|
||||
evaluate: true,
|
||||
keep_fnames: true,
|
||||
passes: 2,
|
||||
reduce_vars: true,
|
||||
side_effects: true,
|
||||
toplevel: true,
|
||||
}
|
||||
input: {
|
||||
var a = 0;
|
||||
if (a)
|
||||
0();
|
||||
else
|
||||
(function f() {
|
||||
f;
|
||||
return a = "PASS";
|
||||
})();
|
||||
console.log(a);
|
||||
}
|
||||
expect: {
|
||||
var a = 0;
|
||||
(a ? 0 : function f() {
|
||||
return a = "PASS";
|
||||
})();
|
||||
console.log(a);
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
issue_5061_1: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
reduce_vars: true,
|
||||
toplevel: true,
|
||||
}
|
||||
input: {
|
||||
var f, a = 1;
|
||||
(f = function() {
|
||||
console.log(a ? "foo" : "bar");
|
||||
})();
|
||||
f(a = 0);
|
||||
}
|
||||
expect: {
|
||||
var f, a = 1;
|
||||
(f = function() {
|
||||
console.log(a ? "foo" : "bar");
|
||||
})();
|
||||
f(a = 0);
|
||||
}
|
||||
expect_stdout: [
|
||||
"foo",
|
||||
"bar",
|
||||
]
|
||||
}
|
||||
|
||||
issue_5061_2: {
|
||||
options = {
|
||||
reduce_vars: true,
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var f, a = 1;
|
||||
(f = function() {
|
||||
console.log(a ? "foo" : "bar");
|
||||
})();
|
||||
f(a = 0);
|
||||
}
|
||||
expect: {
|
||||
var f, a = 1;
|
||||
(f = function() {
|
||||
console.log(a ? "foo" : "bar");
|
||||
})();
|
||||
f(a = 0);
|
||||
}
|
||||
expect_stdout: [
|
||||
"foo",
|
||||
"bar",
|
||||
]
|
||||
}
|
||||
|
||||
issue_5067: {
|
||||
options = {
|
||||
inline: true,
|
||||
reduce_vars: true,
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var f = function() {
|
||||
f();
|
||||
};
|
||||
}
|
||||
expect: {}
|
||||
}
|
||||
|
||||
issue_5096_1: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
passes: 2,
|
||||
reduce_vars: true,
|
||||
side_effects: true,
|
||||
toplevel: true,
|
||||
}
|
||||
input: {
|
||||
var a, b = "FAIL", c = 1;
|
||||
do {
|
||||
a && a();
|
||||
a = function() {
|
||||
b = "PASS";
|
||||
};
|
||||
} while (c--);
|
||||
console.log(b);
|
||||
}
|
||||
expect: {
|
||||
var a, b = "FAIL", c = 1;
|
||||
do {
|
||||
a && a();
|
||||
a = function() {
|
||||
b = "PASS";
|
||||
};
|
||||
} while (c--);
|
||||
console.log(b);
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
issue_5096_2: {
|
||||
options = {
|
||||
passes: 2,
|
||||
reduce_vars: true,
|
||||
side_effects: true,
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var a, b = "FAIL", c = 1;
|
||||
do {
|
||||
a && a();
|
||||
a = function() {
|
||||
b = "PASS";
|
||||
};
|
||||
} while (c--);
|
||||
console.log(b);
|
||||
}
|
||||
expect: {
|
||||
var a, b = "FAIL", c = 1;
|
||||
do {
|
||||
a && a();
|
||||
a = function() {
|
||||
b = "PASS";
|
||||
};
|
||||
} while (c--);
|
||||
console.log(b);
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
issue_5096_3: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
reduce_vars: true,
|
||||
side_effects: true,
|
||||
toplevel: true,
|
||||
}
|
||||
input: {
|
||||
var b = "FAIL", c = 1;
|
||||
do {
|
||||
a && a();
|
||||
var a = function() {
|
||||
b = "PASS";
|
||||
};
|
||||
} while (c--);
|
||||
console.log(b);
|
||||
}
|
||||
expect: {
|
||||
var b = "FAIL", c = 1;
|
||||
do {
|
||||
a && a();
|
||||
var a = function() {
|
||||
b = "PASS";
|
||||
};
|
||||
} while (c--);
|
||||
console.log(b);
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
issue_5096_4: {
|
||||
options = {
|
||||
reduce_vars: true,
|
||||
side_effects: true,
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var b = "FAIL", c = 1;
|
||||
do {
|
||||
a && a();
|
||||
var a = function() {
|
||||
b = "PASS";
|
||||
};
|
||||
} while (c--);
|
||||
console.log(b);
|
||||
}
|
||||
expect: {
|
||||
var b = "FAIL", c = 1;
|
||||
do {
|
||||
a && a();
|
||||
var a = function() {
|
||||
b = "PASS";
|
||||
};
|
||||
} while (c--);
|
||||
console.log(b);
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
issue_5098: {
|
||||
options = {
|
||||
reduce_vars: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
(function(o) {
|
||||
function f() {
|
||||
f = console.log;
|
||||
if (o.p++)
|
||||
throw "FAIL";
|
||||
f("PASS");
|
||||
}
|
||||
return f;
|
||||
})({ p: 0 })();
|
||||
}
|
||||
expect: {
|
||||
(function(o) {
|
||||
function f() {
|
||||
f = console.log;
|
||||
if (o.p++)
|
||||
throw "FAIL";
|
||||
f("PASS");
|
||||
}
|
||||
return f;
|
||||
})({ p: 0 })();
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
shorter_without_void: {
|
||||
options = {
|
||||
inline: true,
|
||||
passes: 2,
|
||||
reduce_vars: true,
|
||||
side_effects: true,
|
||||
toplevel: true,
|
||||
}
|
||||
input: {
|
||||
var a;
|
||||
function f(b) {
|
||||
a = b;
|
||||
}
|
||||
f("foo");
|
||||
console.log(a) || f("bar");
|
||||
console.log(a, f("baz"));
|
||||
console.log(a);
|
||||
}
|
||||
expect: {
|
||||
var a;
|
||||
function f(b) {
|
||||
a = b;
|
||||
}
|
||||
a = "foo";
|
||||
console.log(a) || (a = "bar");
|
||||
console.log(a, f("baz"));
|
||||
console.log(a);
|
||||
}
|
||||
expect_stdout: [
|
||||
"foo",
|
||||
"bar undefined",
|
||||
"baz",
|
||||
]
|
||||
}
|
||||
|
||||
issue_5120: {
|
||||
options = {
|
||||
functions: true,
|
||||
reduce_vars: true,
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var a = function f() {
|
||||
function g() {
|
||||
f || g();
|
||||
}
|
||||
g();
|
||||
return f.valueOf();
|
||||
};
|
||||
console.log(a() === a ? "PASS" : "FAIL");
|
||||
}
|
||||
expect: {
|
||||
function a() {
|
||||
(function g() {
|
||||
a || g();
|
||||
})();
|
||||
return a.valueOf();
|
||||
}
|
||||
console.log(a() === a ? "PASS" : "FAIL");
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
@@ -777,6 +777,32 @@ issue_3046: {
|
||||
}
|
||||
|
||||
issue_3071_1: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
hoist_props: true,
|
||||
inline: true,
|
||||
join_vars: true,
|
||||
passes: 3,
|
||||
reduce_vars: true,
|
||||
sequences: true,
|
||||
side_effects: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
(function() {
|
||||
var obj = {};
|
||||
obj.one = 1;
|
||||
obj.two = 2;
|
||||
console.log(obj.one, obj.two);
|
||||
})();
|
||||
}
|
||||
expect: {
|
||||
console.log(1, 2);
|
||||
}
|
||||
expect_stdout: "1 2"
|
||||
}
|
||||
|
||||
issue_3071_1_toplevel: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
hoist_props: true,
|
||||
@@ -1094,3 +1120,24 @@ object_super: {
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
issue_4985: {
|
||||
options = {
|
||||
hoist_props: true,
|
||||
reduce_vars: true,
|
||||
toplevel: true,
|
||||
}
|
||||
input: {
|
||||
var a = { p: 42 };
|
||||
console.log(function() {
|
||||
a;
|
||||
}());
|
||||
}
|
||||
expect: {
|
||||
var a_p = 42;
|
||||
console.log(function() {
|
||||
({});
|
||||
}());
|
||||
}
|
||||
expect_stdout: "undefined"
|
||||
}
|
||||
|
||||
@@ -140,7 +140,6 @@ issue_4487: {
|
||||
functions: true,
|
||||
hoist_vars: true,
|
||||
keep_fnames: true,
|
||||
passes: 2,
|
||||
reduce_vars: true,
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
@@ -153,7 +152,8 @@ issue_4487: {
|
||||
}
|
||||
expect: {
|
||||
function a() {
|
||||
var a = console.log(typeof a);
|
||||
var f;
|
||||
console.log(typeof f);
|
||||
}
|
||||
a();
|
||||
}
|
||||
@@ -240,3 +240,158 @@ issue_4736: {
|
||||
}
|
||||
expect_stdout: "1073741824"
|
||||
}
|
||||
|
||||
issue_4839: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
hoist_vars: true,
|
||||
keep_fargs: false,
|
||||
reduce_vars: true,
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var o = function(a, b) {
|
||||
return b && b;
|
||||
}("foo");
|
||||
for (var k in o)
|
||||
throw "FAIL";
|
||||
console.log("PASS");
|
||||
}
|
||||
expect: {
|
||||
var k, o = void 0;
|
||||
for (k in o)
|
||||
throw "FAIL";
|
||||
console.log("PASS");
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
issue_4859: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
hoist_vars: true,
|
||||
keep_infinity: true,
|
||||
merge_vars: true,
|
||||
reduce_vars: true,
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
function f(a) {
|
||||
var b = (a = 2, 1 / 0), c = 3;
|
||||
var d = a + b;
|
||||
console.log(d);
|
||||
return f;
|
||||
}
|
||||
f();
|
||||
}
|
||||
expect: {
|
||||
(function f(a) {
|
||||
var d = 1 / 0, d = Infinity;
|
||||
console.log(d);
|
||||
return f;
|
||||
})();
|
||||
}
|
||||
expect_stdout: "Infinity"
|
||||
}
|
||||
|
||||
issue_4893_1: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
evaluate: true,
|
||||
hoist_vars: true,
|
||||
reduce_vars: true,
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
function f() {
|
||||
function g() {}
|
||||
var a = null;
|
||||
var b = null;
|
||||
var c = null;
|
||||
b.p += a = 42;
|
||||
f;
|
||||
}
|
||||
try {
|
||||
f();
|
||||
} catch (e) {
|
||||
console.log("PASS");
|
||||
}
|
||||
}
|
||||
expect: {
|
||||
try{
|
||||
(function f() {
|
||||
null.p += 42;
|
||||
f;
|
||||
})();
|
||||
} catch (e) {
|
||||
console.log("PASS");
|
||||
}
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
issue_4893_2: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
hoist_vars: true,
|
||||
pure_getters: "strict",
|
||||
reduce_vars: true,
|
||||
side_effects: true,
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
function f() {
|
||||
function g() {}
|
||||
var a = null;
|
||||
var b = null;
|
||||
var c = null;
|
||||
b.p += a = 42;
|
||||
f;
|
||||
}
|
||||
try {
|
||||
f();
|
||||
} catch (e) {
|
||||
console.log("PASS");
|
||||
}
|
||||
}
|
||||
expect: {
|
||||
try{
|
||||
(function() {
|
||||
var b;
|
||||
b = null;
|
||||
b.p += 42;
|
||||
})();
|
||||
} catch (e) {
|
||||
console.log("PASS");
|
||||
}
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
issue_4898: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
evaluate: true,
|
||||
hoist_vars: true,
|
||||
loops: true,
|
||||
reduce_vars: true,
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
do {
|
||||
var b = [ console.log("PASS") ];
|
||||
var c = b;
|
||||
} while (c.p = 0);
|
||||
}
|
||||
expect: {
|
||||
var b;
|
||||
b = [ console.log("PASS") ];
|
||||
b.p = 0;
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
File diff suppressed because it is too large
Load Diff
@@ -23,7 +23,7 @@ keys_only: {
|
||||
input: {
|
||||
import { as as foo, bar, delete as baz } from "moo";
|
||||
}
|
||||
expect_exact: 'import{as as foo,bar as bar,delete as baz}from"moo";'
|
||||
expect_exact: 'import{as as foo,bar,delete as baz}from"moo";'
|
||||
}
|
||||
|
||||
default_all: {
|
||||
@@ -37,7 +37,7 @@ default_keys: {
|
||||
input: {
|
||||
import foo, { bar } from "baz";
|
||||
}
|
||||
expect_exact: 'import foo,{bar as bar}from"baz";'
|
||||
expect_exact: 'import foo,{bar}from"baz";'
|
||||
}
|
||||
|
||||
dynamic: {
|
||||
@@ -54,13 +54,22 @@ dynamic_nought: {
|
||||
expect_exact: "import(foo);"
|
||||
}
|
||||
|
||||
import_meta: {
|
||||
import_meta_1: {
|
||||
input: {
|
||||
console.log(import.meta, import.meta.url);
|
||||
}
|
||||
expect_exact: "console.log(import.meta,import.meta.url);"
|
||||
}
|
||||
|
||||
import_meta_2: {
|
||||
input: {
|
||||
import.meta.url.split("/").forEach(function(part, index) {
|
||||
console.log(index, part);
|
||||
});
|
||||
}
|
||||
expect_exact: 'import.meta.url.split("/").forEach(function(part,index){console.log(index,part)});'
|
||||
}
|
||||
|
||||
same_quotes: {
|
||||
beautify = {
|
||||
beautify: true,
|
||||
@@ -85,13 +94,15 @@ drop_unused: {
|
||||
}
|
||||
input: {
|
||||
import a, * as b from "foo";
|
||||
import { c, bar as d } from "baz";
|
||||
console.log(c);
|
||||
import { c } from "bar";
|
||||
import { d, _ as e } from "baz";
|
||||
console.log(d);
|
||||
}
|
||||
expect: {
|
||||
import "foo";
|
||||
import { c as c } from "baz";
|
||||
console.log(c);
|
||||
import "bar";
|
||||
import { d as d } from "baz";
|
||||
console.log(d);
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
@@ -4,7 +4,7 @@ mangle_keep_fnames_false: {
|
||||
keep_fnames: true,
|
||||
}
|
||||
mangle = {
|
||||
keep_fnames : false,
|
||||
keep_fnames: false,
|
||||
}
|
||||
input: {
|
||||
"use strict";
|
||||
@@ -30,7 +30,7 @@ mangle_keep_fnames_true: {
|
||||
keep_fnames: true,
|
||||
}
|
||||
mangle = {
|
||||
keep_fnames : true,
|
||||
keep_fnames: true,
|
||||
}
|
||||
input: {
|
||||
"use strict";
|
||||
|
||||
@@ -1,5 +1,6 @@
|
||||
pure_function_calls: {
|
||||
options = {
|
||||
annotations: true,
|
||||
booleans: true,
|
||||
comparisons: true,
|
||||
conditionals: true,
|
||||
@@ -60,6 +61,7 @@ pure_function_calls: {
|
||||
|
||||
pure_function_calls_toplevel: {
|
||||
options = {
|
||||
annotations: true,
|
||||
booleans: true,
|
||||
comparisons: true,
|
||||
conditionals: true,
|
||||
@@ -126,6 +128,7 @@ pure_function_calls_toplevel: {
|
||||
|
||||
should_warn: {
|
||||
options = {
|
||||
annotations: true,
|
||||
booleans: true,
|
||||
conditionals: true,
|
||||
evaluate: true,
|
||||
|
||||
@@ -24,7 +24,7 @@ typeof_eq_undefined: {
|
||||
typeof_eq_undefined_ie8: {
|
||||
options = {
|
||||
comparisons: true,
|
||||
ie8: true,
|
||||
ie: true,
|
||||
typeofs: true,
|
||||
}
|
||||
input: {
|
||||
|
||||
@@ -1,9 +1,9 @@
|
||||
screw_ie8: {
|
||||
options = {
|
||||
ie8: false,
|
||||
ie: false,
|
||||
}
|
||||
mangle = {
|
||||
ie8: false,
|
||||
ie: false,
|
||||
}
|
||||
input: {
|
||||
try { throw "foo"; } catch (x) { console.log(x); }
|
||||
@@ -16,10 +16,10 @@ screw_ie8: {
|
||||
|
||||
support_ie8: {
|
||||
options = {
|
||||
ie8: true,
|
||||
ie: true,
|
||||
}
|
||||
mangle = {
|
||||
ie8: true,
|
||||
ie: true,
|
||||
}
|
||||
input: {
|
||||
try { throw "foo"; } catch (x) { console.log(x); }
|
||||
|
||||
@@ -1,10 +1,10 @@
|
||||
mangle_catch: {
|
||||
options = {
|
||||
ie8: false,
|
||||
ie: false,
|
||||
toplevel: false,
|
||||
}
|
||||
mangle = {
|
||||
ie8: false,
|
||||
ie: false,
|
||||
toplevel: false,
|
||||
}
|
||||
input: {
|
||||
@@ -22,11 +22,11 @@ mangle_catch: {
|
||||
|
||||
mangle_catch_ie8: {
|
||||
options = {
|
||||
ie8: true,
|
||||
ie: true,
|
||||
toplevel: false,
|
||||
}
|
||||
mangle = {
|
||||
ie8: true,
|
||||
ie: true,
|
||||
toplevel: false,
|
||||
}
|
||||
input: {
|
||||
@@ -44,11 +44,11 @@ mangle_catch_ie8: {
|
||||
|
||||
mangle_catch_var: {
|
||||
options = {
|
||||
ie8: false,
|
||||
ie: false,
|
||||
toplevel: false,
|
||||
}
|
||||
mangle = {
|
||||
ie8: false,
|
||||
ie: false,
|
||||
toplevel: false,
|
||||
}
|
||||
input: {
|
||||
@@ -66,11 +66,11 @@ mangle_catch_var: {
|
||||
|
||||
mangle_catch_var_ie8: {
|
||||
options = {
|
||||
ie8: true,
|
||||
ie: true,
|
||||
toplevel: false,
|
||||
}
|
||||
mangle = {
|
||||
ie8: true,
|
||||
ie: true,
|
||||
toplevel: false,
|
||||
}
|
||||
input: {
|
||||
@@ -88,11 +88,11 @@ mangle_catch_var_ie8: {
|
||||
|
||||
mangle_catch_toplevel: {
|
||||
options = {
|
||||
ie8: false,
|
||||
ie: false,
|
||||
toplevel: true,
|
||||
}
|
||||
mangle = {
|
||||
ie8: false,
|
||||
ie: false,
|
||||
toplevel: true,
|
||||
}
|
||||
input: {
|
||||
@@ -110,11 +110,11 @@ mangle_catch_toplevel: {
|
||||
|
||||
mangle_catch_ie8_toplevel: {
|
||||
options = {
|
||||
ie8: true,
|
||||
ie: true,
|
||||
toplevel: true,
|
||||
}
|
||||
mangle = {
|
||||
ie8: true,
|
||||
ie: true,
|
||||
toplevel: true,
|
||||
}
|
||||
input: {
|
||||
@@ -132,11 +132,11 @@ mangle_catch_ie8_toplevel: {
|
||||
|
||||
mangle_catch_var_toplevel: {
|
||||
options = {
|
||||
ie8: false,
|
||||
ie: false,
|
||||
toplevel: true,
|
||||
}
|
||||
mangle = {
|
||||
ie8: false,
|
||||
ie: false,
|
||||
toplevel: true,
|
||||
}
|
||||
input: {
|
||||
@@ -154,11 +154,11 @@ mangle_catch_var_toplevel: {
|
||||
|
||||
mangle_catch_var_ie8_toplevel: {
|
||||
options = {
|
||||
ie8: true,
|
||||
ie: true,
|
||||
toplevel: true,
|
||||
}
|
||||
mangle = {
|
||||
ie8: true,
|
||||
ie: true,
|
||||
toplevel: true,
|
||||
}
|
||||
input: {
|
||||
@@ -176,11 +176,11 @@ mangle_catch_var_ie8_toplevel: {
|
||||
|
||||
mangle_catch_redef_1: {
|
||||
options = {
|
||||
ie8: false,
|
||||
ie: false,
|
||||
toplevel: false,
|
||||
}
|
||||
mangle = {
|
||||
ie8: false,
|
||||
ie: false,
|
||||
toplevel: false,
|
||||
}
|
||||
input: {
|
||||
@@ -198,11 +198,11 @@ mangle_catch_redef_1: {
|
||||
|
||||
mangle_catch_redef_1_ie8: {
|
||||
options = {
|
||||
ie8: true,
|
||||
ie: true,
|
||||
toplevel: false,
|
||||
}
|
||||
mangle = {
|
||||
ie8: true,
|
||||
ie: true,
|
||||
toplevel: false,
|
||||
}
|
||||
input: {
|
||||
@@ -220,11 +220,11 @@ mangle_catch_redef_1_ie8: {
|
||||
|
||||
mangle_catch_redef_1_toplevel: {
|
||||
options = {
|
||||
ie8: false,
|
||||
ie: false,
|
||||
toplevel: true,
|
||||
}
|
||||
mangle = {
|
||||
ie8: false,
|
||||
ie: false,
|
||||
toplevel: true,
|
||||
}
|
||||
input: {
|
||||
@@ -242,11 +242,11 @@ mangle_catch_redef_1_toplevel: {
|
||||
|
||||
mangle_catch_redef_1_ie8_toplevel: {
|
||||
options = {
|
||||
ie8: true,
|
||||
ie: true,
|
||||
toplevel: true,
|
||||
}
|
||||
mangle = {
|
||||
ie8: true,
|
||||
ie: true,
|
||||
toplevel: true,
|
||||
}
|
||||
input: {
|
||||
@@ -264,11 +264,11 @@ mangle_catch_redef_1_ie8_toplevel: {
|
||||
|
||||
mangle_catch_redef_2: {
|
||||
options = {
|
||||
ie8: false,
|
||||
ie: false,
|
||||
toplevel: false,
|
||||
}
|
||||
mangle = {
|
||||
ie8: false,
|
||||
ie: false,
|
||||
toplevel: false,
|
||||
}
|
||||
input: {
|
||||
@@ -285,11 +285,11 @@ mangle_catch_redef_2: {
|
||||
|
||||
mangle_catch_redef_2_ie8: {
|
||||
options = {
|
||||
ie8: true,
|
||||
ie: true,
|
||||
toplevel: false,
|
||||
}
|
||||
mangle = {
|
||||
ie8: true,
|
||||
ie: true,
|
||||
toplevel: false,
|
||||
}
|
||||
input: {
|
||||
@@ -306,11 +306,11 @@ mangle_catch_redef_2_ie8: {
|
||||
|
||||
mangle_catch_redef_2_toplevel: {
|
||||
options = {
|
||||
ie8: false,
|
||||
ie: false,
|
||||
toplevel: true,
|
||||
}
|
||||
mangle = {
|
||||
ie8: false,
|
||||
ie: false,
|
||||
toplevel: true,
|
||||
}
|
||||
input: {
|
||||
@@ -327,11 +327,11 @@ mangle_catch_redef_2_toplevel: {
|
||||
|
||||
mangle_catch_redef_2_ie8_toplevel: {
|
||||
options = {
|
||||
ie8: true,
|
||||
ie: true,
|
||||
toplevel: true,
|
||||
}
|
||||
mangle = {
|
||||
ie8: true,
|
||||
ie: true,
|
||||
toplevel: true,
|
||||
}
|
||||
input: {
|
||||
@@ -348,7 +348,7 @@ mangle_catch_redef_2_ie8_toplevel: {
|
||||
|
||||
mangle_catch_redef_3: {
|
||||
mangle = {
|
||||
ie8: false,
|
||||
ie: false,
|
||||
toplevel: false,
|
||||
}
|
||||
input: {
|
||||
@@ -371,7 +371,7 @@ mangle_catch_redef_3: {
|
||||
|
||||
mangle_catch_redef_3_toplevel: {
|
||||
mangle = {
|
||||
ie8: false,
|
||||
ie: false,
|
||||
toplevel: true,
|
||||
}
|
||||
input: {
|
||||
@@ -394,7 +394,7 @@ mangle_catch_redef_3_toplevel: {
|
||||
|
||||
mangle_catch_redef_3_ie8: {
|
||||
mangle = {
|
||||
ie8: true,
|
||||
ie: true,
|
||||
toplevel: false,
|
||||
}
|
||||
input: {
|
||||
@@ -417,7 +417,7 @@ mangle_catch_redef_3_ie8: {
|
||||
|
||||
mangle_catch_redef_3_ie8_toplevel: {
|
||||
mangle = {
|
||||
ie8: true,
|
||||
ie: true,
|
||||
toplevel: true,
|
||||
}
|
||||
input: {
|
||||
|
||||
@@ -1,6 +1,6 @@
|
||||
function_iife_catch: {
|
||||
mangle = {
|
||||
ie8: false,
|
||||
ie: false,
|
||||
}
|
||||
input: {
|
||||
function f(n) {
|
||||
@@ -21,7 +21,7 @@ function_iife_catch: {
|
||||
|
||||
function_iife_catch_ie8: {
|
||||
mangle = {
|
||||
ie8: true,
|
||||
ie: true,
|
||||
}
|
||||
input: {
|
||||
function f(n) {
|
||||
@@ -42,7 +42,7 @@ function_iife_catch_ie8: {
|
||||
|
||||
function_catch_catch: {
|
||||
mangle = {
|
||||
ie8: false,
|
||||
ie: false,
|
||||
}
|
||||
input: {
|
||||
var o = 0;
|
||||
@@ -70,7 +70,7 @@ function_catch_catch: {
|
||||
|
||||
function_catch_catch_ie8: {
|
||||
mangle = {
|
||||
ie8: true,
|
||||
ie: true,
|
||||
}
|
||||
input: {
|
||||
var o = 0;
|
||||
|
||||
@@ -426,6 +426,7 @@ wrap_iife_in_return_call: {
|
||||
|
||||
pure_annotation_1: {
|
||||
options = {
|
||||
annotations: true,
|
||||
inline: true,
|
||||
side_effects: true,
|
||||
}
|
||||
@@ -439,6 +440,7 @@ pure_annotation_1: {
|
||||
|
||||
pure_annotation_2: {
|
||||
options = {
|
||||
annotations: true,
|
||||
collapse_vars: true,
|
||||
inline: true,
|
||||
side_effects: true,
|
||||
|
||||
@@ -1015,7 +1015,7 @@ issue_3856: {
|
||||
console.log(function() {
|
||||
(function() {
|
||||
var a, b;
|
||||
if (a) return !!a;
|
||||
if (a) return a, 1;
|
||||
for (a = 0; !console;);
|
||||
return 0;
|
||||
})();
|
||||
@@ -1024,7 +1024,7 @@ issue_3856: {
|
||||
expect_stdout: "undefined"
|
||||
}
|
||||
|
||||
issue_3916: {
|
||||
issue_3916_1: {
|
||||
options = {
|
||||
join_vars: true,
|
||||
}
|
||||
@@ -1044,8 +1044,8 @@ issue_3916: {
|
||||
var o = {
|
||||
p: "PASS",
|
||||
__proto__: 42,
|
||||
q: "FAIL",
|
||||
};
|
||||
o.q = "FAIL";
|
||||
o.__proto__ = {
|
||||
p: "FAIL",
|
||||
q: "PASS",
|
||||
@@ -1056,6 +1056,62 @@ issue_3916: {
|
||||
expect_stdout: "object PASS true PASS"
|
||||
}
|
||||
|
||||
issue_3916_2: {
|
||||
options = {
|
||||
join_vars: true,
|
||||
}
|
||||
input: {
|
||||
var log = console.log, o = {};
|
||||
o.p = "FAIL 1";
|
||||
o.__proto__ = {
|
||||
get p() {
|
||||
return "FAIL 2";
|
||||
},
|
||||
set p(u) {
|
||||
log("FAIL 3");
|
||||
},
|
||||
set q(v) {
|
||||
log("PASS 1");
|
||||
},
|
||||
get q() {
|
||||
return "PASS 3";
|
||||
},
|
||||
};
|
||||
o.p = "PASS 2";
|
||||
o.q = "FAIL 4";
|
||||
log(o.p);
|
||||
log(o.q);
|
||||
}
|
||||
expect: {
|
||||
var log = console.log, o = {
|
||||
p: "FAIL 1",
|
||||
__proto__: {
|
||||
get p() {
|
||||
return "FAIL 2";
|
||||
},
|
||||
set p(u) {
|
||||
log("FAIL 3");
|
||||
},
|
||||
set q(v) {
|
||||
log("PASS 1");
|
||||
},
|
||||
get q() {
|
||||
return "PASS 3";
|
||||
},
|
||||
},
|
||||
};
|
||||
o.p = "PASS 2";
|
||||
o.q = "FAIL 4";
|
||||
log(o.p);
|
||||
log(o.q);
|
||||
}
|
||||
expect_stdout: [
|
||||
"PASS 1",
|
||||
"PASS 2",
|
||||
"PASS 3",
|
||||
]
|
||||
}
|
||||
|
||||
assign_var: {
|
||||
options = {
|
||||
join_vars: true,
|
||||
|
||||
@@ -117,6 +117,7 @@ issue_1858: {
|
||||
collapse_vars: true,
|
||||
keep_fargs: false,
|
||||
pure_getters: true,
|
||||
reduce_vars: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
@@ -950,7 +951,7 @@ function_name_mangle_ie8: {
|
||||
unused: true,
|
||||
}
|
||||
mangle = {
|
||||
ie8: true,
|
||||
ie: true,
|
||||
toplevel: true,
|
||||
}
|
||||
input: {
|
||||
@@ -1157,8 +1158,8 @@ replace_all_var_scope: {
|
||||
var a = 100, b = 10;
|
||||
(function(r, a) {
|
||||
switch (~a) {
|
||||
case (b += a):
|
||||
case a++:
|
||||
case (b += a):
|
||||
case a++:
|
||||
}
|
||||
})(--b, a);
|
||||
console.log(a, b);
|
||||
@@ -1167,8 +1168,8 @@ replace_all_var_scope: {
|
||||
var a = 100, b = 10;
|
||||
(function(c) {
|
||||
switch (~a) {
|
||||
case (b += a):
|
||||
case c++:
|
||||
case (b += a):
|
||||
case c++:
|
||||
}
|
||||
})((--b, a));
|
||||
console.log(a, b);
|
||||
|
||||
@@ -20,6 +20,39 @@ retain_block: {
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
retain_assignment: {
|
||||
options = {
|
||||
dead_code: true,
|
||||
reduce_vars: true,
|
||||
}
|
||||
input: {
|
||||
"use strict";
|
||||
function f() {
|
||||
return a = 0;
|
||||
let a;
|
||||
}
|
||||
try {
|
||||
f();
|
||||
} catch (e) {
|
||||
console.log("PASS");
|
||||
}
|
||||
}
|
||||
expect: {
|
||||
"use strict";
|
||||
function f() {
|
||||
return a = 0;
|
||||
let a;
|
||||
}
|
||||
try {
|
||||
f();
|
||||
} catch (e) {
|
||||
console.log("PASS");
|
||||
}
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
retain_catch: {
|
||||
options = {
|
||||
dead_code: true,
|
||||
@@ -359,7 +392,7 @@ reduce_block_2_toplevel: {
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
reduce_vars: {
|
||||
reduce_vars_1: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
reduce_vars: true,
|
||||
@@ -381,6 +414,121 @@ reduce_vars: {
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
reduce_vars_2: {
|
||||
options = {
|
||||
reduce_funcs: true,
|
||||
reduce_vars: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
"use strict";
|
||||
(function() {
|
||||
function f() {
|
||||
console.log(typeof a);
|
||||
}
|
||||
for (let a in [ 42 ])
|
||||
f();
|
||||
})();
|
||||
}
|
||||
expect: {
|
||||
"use strict";
|
||||
(function() {
|
||||
function f() {
|
||||
console.log(typeof a);
|
||||
}
|
||||
for (let a in [ 42 ])
|
||||
f();
|
||||
})();
|
||||
}
|
||||
expect_stdout: "undefined"
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
reduce_vars_3: {
|
||||
options = {
|
||||
reduce_funcs: true,
|
||||
reduce_vars: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
"use strict";
|
||||
(function(a) {
|
||||
let i = 1;
|
||||
function f() {
|
||||
i = 0;
|
||||
}
|
||||
for (let i = 0, x = 0; i < a.length; i++, x++) {
|
||||
if (x != i) {
|
||||
console.log("FAIL");
|
||||
break;
|
||||
}
|
||||
f();
|
||||
console.log(a[i]);
|
||||
}
|
||||
console.log(i);
|
||||
})([ 4, 2 ]);
|
||||
}
|
||||
expect: {
|
||||
"use strict";
|
||||
(function(a) {
|
||||
let i = 1;
|
||||
function f() {
|
||||
i = 0;
|
||||
}
|
||||
for (let i = 0, x = 0; i < a.length; i++, x++) {
|
||||
if (x != i) {
|
||||
console.log("FAIL");
|
||||
break;
|
||||
}
|
||||
f();
|
||||
console.log(a[i]);
|
||||
}
|
||||
console.log(i);
|
||||
})([ 4, 2 ]);
|
||||
}
|
||||
expect_stdout: [
|
||||
"4",
|
||||
"2",
|
||||
"0",
|
||||
]
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
reduce_lambda: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
functions: true,
|
||||
reduce_vars: true,
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
"use strict";
|
||||
let f = function() {
|
||||
console.log(a, b);
|
||||
};
|
||||
let a = "foo", b = 42;
|
||||
f();
|
||||
b = "bar";
|
||||
f();
|
||||
}
|
||||
expect: {
|
||||
"use strict";
|
||||
function f() {
|
||||
console.log("foo", b);
|
||||
}
|
||||
let b = 42;
|
||||
f();
|
||||
b = "bar";
|
||||
f();
|
||||
}
|
||||
expect_stdout: [
|
||||
"foo 42",
|
||||
"foo bar",
|
||||
]
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
hoist_props: {
|
||||
options = {
|
||||
hoist_props: true,
|
||||
@@ -456,6 +604,38 @@ loop_block_2: {
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
do_break: {
|
||||
options = {
|
||||
loops: true,
|
||||
}
|
||||
input: {
|
||||
"use strict";
|
||||
try {
|
||||
do {
|
||||
if (a)
|
||||
break;
|
||||
let a;
|
||||
} while (!console);
|
||||
} catch (e) {
|
||||
console.log("PASS");
|
||||
}
|
||||
}
|
||||
expect: {
|
||||
"use strict";
|
||||
try {
|
||||
do {
|
||||
if (a)
|
||||
break;
|
||||
let a;
|
||||
} while (!console);
|
||||
} catch (e) {
|
||||
console.log("PASS");
|
||||
}
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
do_continue: {
|
||||
options = {
|
||||
loops: true,
|
||||
@@ -516,6 +696,82 @@ dead_block_after_return: {
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
if_return_1: {
|
||||
options = {
|
||||
if_return: true,
|
||||
}
|
||||
input: {
|
||||
"use strict";
|
||||
function f(a) {
|
||||
function g() {
|
||||
return b = "PASS";
|
||||
}
|
||||
if (a)
|
||||
return g();
|
||||
let b;
|
||||
return g();
|
||||
};
|
||||
console.log(f());
|
||||
}
|
||||
expect: {
|
||||
"use strict";
|
||||
function f(a) {
|
||||
function g() {
|
||||
return b = "PASS";
|
||||
}
|
||||
if (a)
|
||||
return g();
|
||||
let b;
|
||||
return g();
|
||||
};
|
||||
console.log(f());
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
if_return_2: {
|
||||
options = {
|
||||
if_return: true,
|
||||
}
|
||||
input: {
|
||||
"use strict";
|
||||
function f(a) {
|
||||
function g() {
|
||||
return b = "FAIL";
|
||||
}
|
||||
if (a)
|
||||
return g();
|
||||
let b;
|
||||
return g();
|
||||
};
|
||||
try {
|
||||
console.log(f(42));
|
||||
} catch (e) {
|
||||
console.log("PASS");
|
||||
}
|
||||
}
|
||||
expect: {
|
||||
"use strict";
|
||||
function f(a) {
|
||||
function g() {
|
||||
return b = "FAIL";
|
||||
}
|
||||
if (a)
|
||||
return g();
|
||||
let b;
|
||||
return g();
|
||||
};
|
||||
try {
|
||||
console.log(f(42));
|
||||
} catch (e) {
|
||||
console.log("PASS");
|
||||
}
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
do_if_continue_1: {
|
||||
options = {
|
||||
if_return: true,
|
||||
@@ -817,6 +1073,7 @@ issue_4210: {
|
||||
issue_4212_1: {
|
||||
options = {
|
||||
dead_code: true,
|
||||
reduce_vars: true,
|
||||
}
|
||||
input: {
|
||||
"use strict";
|
||||
@@ -907,7 +1164,7 @@ issue_4225: {
|
||||
|
||||
issue_4229: {
|
||||
options = {
|
||||
ie8: true,
|
||||
ie: true,
|
||||
side_effects: true,
|
||||
}
|
||||
input: {
|
||||
@@ -938,7 +1195,7 @@ issue_4229: {
|
||||
|
||||
issue_4231: {
|
||||
options = {
|
||||
ie8: true,
|
||||
ie: true,
|
||||
side_effects: true,
|
||||
}
|
||||
input: {
|
||||
@@ -1307,7 +1564,7 @@ issue_4438: {
|
||||
|
||||
issue_4531_1: {
|
||||
mangle = {
|
||||
ie8: true,
|
||||
ie: true,
|
||||
toplevel: true,
|
||||
}
|
||||
input: {
|
||||
@@ -1333,11 +1590,11 @@ issue_4531_1: {
|
||||
issue_4531_2: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
ie8: true,
|
||||
ie: true,
|
||||
toplevel: true,
|
||||
}
|
||||
mangle = {
|
||||
ie8: true,
|
||||
ie: true,
|
||||
toplevel: true,
|
||||
}
|
||||
input: {
|
||||
@@ -1426,3 +1683,67 @@ issue_4691: {
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
issue_4848: {
|
||||
options = {
|
||||
if_return: true,
|
||||
}
|
||||
input: {
|
||||
"use strict";
|
||||
function f(a) {
|
||||
a(function() {
|
||||
console.log(b);
|
||||
});
|
||||
if (!console)
|
||||
return;
|
||||
let b = "PASS";
|
||||
}
|
||||
var g;
|
||||
f(function(h) {
|
||||
g = h;
|
||||
});
|
||||
g();
|
||||
}
|
||||
expect: {
|
||||
"use strict";
|
||||
function f(a) {
|
||||
a(function() {
|
||||
console.log(b);
|
||||
});
|
||||
if (!console)
|
||||
return;
|
||||
let b = "PASS";
|
||||
}
|
||||
var g;
|
||||
f(function(h) {
|
||||
g = h;
|
||||
});
|
||||
g();
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
issue_4985: {
|
||||
options = {
|
||||
hoist_props: true,
|
||||
reduce_vars: true,
|
||||
toplevel: true,
|
||||
}
|
||||
input: {
|
||||
"use strict";
|
||||
let a = { p: 42 };
|
||||
console.log(function() {
|
||||
a;
|
||||
}());
|
||||
}
|
||||
expect: {
|
||||
"use strict";
|
||||
let a = { p: 42 };
|
||||
console.log(function() {
|
||||
a;
|
||||
}());
|
||||
}
|
||||
expect_stdout: "undefined"
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
@@ -193,9 +193,9 @@ evaluate: {
|
||||
} while (false);
|
||||
}
|
||||
expect: {
|
||||
for(;;)
|
||||
for (;;)
|
||||
a();
|
||||
for(;;)
|
||||
for (;;)
|
||||
c();
|
||||
d();
|
||||
}
|
||||
@@ -265,7 +265,7 @@ issue_1532_2: {
|
||||
issue_186: {
|
||||
beautify = {
|
||||
beautify: false,
|
||||
ie8: false,
|
||||
ie: false,
|
||||
}
|
||||
input: {
|
||||
var x = 3;
|
||||
@@ -284,7 +284,7 @@ issue_186: {
|
||||
issue_186_ie8: {
|
||||
beautify = {
|
||||
beautify: false,
|
||||
ie8: true,
|
||||
ie: true,
|
||||
}
|
||||
input: {
|
||||
var x = 3;
|
||||
@@ -303,7 +303,7 @@ issue_186_ie8: {
|
||||
issue_186_beautify: {
|
||||
beautify = {
|
||||
beautify: true,
|
||||
ie8: false,
|
||||
ie: false,
|
||||
}
|
||||
input: {
|
||||
var x = 3;
|
||||
@@ -330,7 +330,7 @@ issue_186_beautify: {
|
||||
issue_186_beautify_ie8: {
|
||||
beautify = {
|
||||
beautify: true,
|
||||
ie8: true,
|
||||
ie: true,
|
||||
}
|
||||
input: {
|
||||
var x = 3;
|
||||
@@ -360,7 +360,7 @@ issue_186_braces: {
|
||||
beautify = {
|
||||
beautify: false,
|
||||
braces: true,
|
||||
ie8: false,
|
||||
ie: false,
|
||||
}
|
||||
input: {
|
||||
var x = 3;
|
||||
@@ -380,7 +380,7 @@ issue_186_braces_ie8: {
|
||||
beautify = {
|
||||
beautify: false,
|
||||
braces: true,
|
||||
ie8: true,
|
||||
ie: true,
|
||||
}
|
||||
input: {
|
||||
var x = 3;
|
||||
@@ -400,7 +400,7 @@ issue_186_beautify_braces: {
|
||||
beautify = {
|
||||
beautify: true,
|
||||
braces: true,
|
||||
ie8: false,
|
||||
ie: false,
|
||||
}
|
||||
input: {
|
||||
var x = 3;
|
||||
@@ -432,7 +432,7 @@ issue_186_beautify_braces_ie8: {
|
||||
beautify = {
|
||||
beautify: true,
|
||||
braces: true,
|
||||
ie8: true,
|
||||
ie: true,
|
||||
}
|
||||
input: {
|
||||
var x = 3;
|
||||
@@ -829,6 +829,18 @@ empty_for_in_prop_init: {
|
||||
}
|
||||
|
||||
for_of: {
|
||||
input: {
|
||||
var a = [ "PASS", 42 ];
|
||||
a.p = "FAIL";
|
||||
for (a of (null, a))
|
||||
console.log(a);
|
||||
}
|
||||
expect_exact: 'var a=["PASS",42];a.p="FAIL";for(a of(null,a))console.log(a);'
|
||||
expect_stdout: true
|
||||
node_version: ">=0.12"
|
||||
}
|
||||
|
||||
for_async_of: {
|
||||
input: {
|
||||
var async = [ "PASS", 42 ];
|
||||
async.p = "FAIL";
|
||||
@@ -836,11 +848,22 @@ for_of: {
|
||||
console.log(async);
|
||||
}
|
||||
expect_exact: 'var async=["PASS",42];async.p="FAIL";for(async of(null,async))console.log(async);'
|
||||
expect_stdout: [
|
||||
"PASS",
|
||||
"42",
|
||||
]
|
||||
node_version: ">=0.12"
|
||||
expect_stdout: true
|
||||
node_version: ">=0.12 <16"
|
||||
}
|
||||
|
||||
for_of_regexp: {
|
||||
input: {
|
||||
for (var a of /foo/);
|
||||
}
|
||||
expect_exact: "for(var a of/foo/);"
|
||||
}
|
||||
|
||||
for_await_of_regexp: {
|
||||
input: {
|
||||
for await (var a of /foo/);
|
||||
}
|
||||
expect_exact: "for await(var a of/foo/);"
|
||||
}
|
||||
|
||||
issue_3631_1: {
|
||||
|
||||
@@ -359,7 +359,7 @@ issue_4107: {
|
||||
|
||||
issue_4109: {
|
||||
options = {
|
||||
ie8: true,
|
||||
ie: true,
|
||||
merge_vars: true,
|
||||
toplevel: true,
|
||||
}
|
||||
@@ -2735,8 +2735,7 @@ issue_4135: {
|
||||
0;
|
||||
a++;
|
||||
if (!a)
|
||||
c = (a++, c = 0, void (c && c.p));
|
||||
var c;
|
||||
var c = void a++;
|
||||
console.log(a, -1, c);
|
||||
}
|
||||
expect_stdout: "1 -1 undefined"
|
||||
@@ -3075,7 +3074,7 @@ issue_4237_2: {
|
||||
console.log(function(a) {
|
||||
do {
|
||||
switch (0) {
|
||||
case 0:
|
||||
default:
|
||||
var b = a++;
|
||||
if (b)
|
||||
return "FAIL";
|
||||
@@ -3301,3 +3300,80 @@ issue_4761: {
|
||||
}
|
||||
expect_stdout: "undefined"
|
||||
}
|
||||
|
||||
issue_4956_1: {
|
||||
options = {
|
||||
merge_vars: true,
|
||||
toplevel: true,
|
||||
}
|
||||
input: {
|
||||
var a, b;
|
||||
function f(c) {
|
||||
switch (c) {
|
||||
case 0:
|
||||
a = { p: 42 };
|
||||
|
||||
case 1:
|
||||
b = a.p;
|
||||
console.log(b);
|
||||
}
|
||||
}
|
||||
f(0);
|
||||
f(1);
|
||||
}
|
||||
expect: {
|
||||
var a, b;
|
||||
function f(c) {
|
||||
switch (c) {
|
||||
case 0:
|
||||
a = { p: 42 };
|
||||
|
||||
case 1:
|
||||
b = a.p;
|
||||
console.log(b);
|
||||
}
|
||||
}
|
||||
f(0);
|
||||
f(1);
|
||||
}
|
||||
expect_stdout: [
|
||||
"42",
|
||||
"42",
|
||||
]
|
||||
}
|
||||
|
||||
issue_4956_2: {
|
||||
options = {
|
||||
merge_vars: true,
|
||||
toplevel: true,
|
||||
}
|
||||
input: {
|
||||
var a, b;
|
||||
function f(c) {
|
||||
if (0 == c) {
|
||||
console;
|
||||
a = { p: 42 };
|
||||
}
|
||||
b = a.p;
|
||||
if (1 == c)
|
||||
console.log(b);
|
||||
}
|
||||
f(0);
|
||||
f(1);
|
||||
}
|
||||
expect: {
|
||||
var a, b;
|
||||
function f(c) {
|
||||
if (0 == c) {
|
||||
console;
|
||||
a = { p: 42 };
|
||||
}
|
||||
b = a.p;
|
||||
if (1 == c)
|
||||
console.log(b);
|
||||
}
|
||||
f(0);
|
||||
f(1);
|
||||
}
|
||||
expect_stdout: "42"
|
||||
}
|
||||
|
||||
@@ -110,10 +110,162 @@ conditional_assignment_4: {
|
||||
node_version: ">=14"
|
||||
}
|
||||
|
||||
de_morgan_1: {
|
||||
options = {
|
||||
booleans: true,
|
||||
}
|
||||
input: {
|
||||
function f(a) {
|
||||
return a ?? a;
|
||||
}
|
||||
console.log(f(null), f(42));
|
||||
}
|
||||
expect: {
|
||||
function f(a) {
|
||||
return a;
|
||||
}
|
||||
console.log(f(null), f(42));
|
||||
}
|
||||
expect_stdout: "null 42"
|
||||
node_version: ">=14"
|
||||
}
|
||||
|
||||
de_morgan_2a: {
|
||||
options = {
|
||||
booleans: true,
|
||||
conditionals: true,
|
||||
evaluate: true,
|
||||
}
|
||||
input: {
|
||||
function f(a, b) {
|
||||
return a || (a ?? b);
|
||||
}
|
||||
console.log(f(null), f(null, {}));
|
||||
console.log(f(42), f(42, {}));
|
||||
}
|
||||
expect: {
|
||||
function f(a, b) {
|
||||
return a || (a ?? b);
|
||||
}
|
||||
console.log(f(null), f(null, {}));
|
||||
console.log(f(42), f(42, {}));
|
||||
}
|
||||
expect_stdout: [
|
||||
"undefined {}",
|
||||
"42 42",
|
||||
]
|
||||
node_version: ">=14"
|
||||
}
|
||||
|
||||
de_morgan_2b: {
|
||||
options = {
|
||||
booleans: true,
|
||||
evaluate: true,
|
||||
}
|
||||
input: {
|
||||
function f(a, b) {
|
||||
return a && (a ?? b);
|
||||
}
|
||||
console.log(f(null), f(null, {}));
|
||||
console.log(f(42), f(42, {}));
|
||||
}
|
||||
expect: {
|
||||
function f(a, b) {
|
||||
return a;
|
||||
}
|
||||
console.log(f(null), f(null, {}));
|
||||
console.log(f(42), f(42, {}));
|
||||
}
|
||||
expect_stdout: [
|
||||
"null null",
|
||||
"42 42",
|
||||
]
|
||||
node_version: ">=14"
|
||||
}
|
||||
|
||||
de_morgan_2c: {
|
||||
options = {
|
||||
booleans: true,
|
||||
evaluate: true,
|
||||
side_effects: true,
|
||||
}
|
||||
input: {
|
||||
function f(a, b) {
|
||||
return a ?? (a || b);
|
||||
}
|
||||
console.log(f(null), f(null, {}));
|
||||
console.log(f(42), f(42, {}));
|
||||
}
|
||||
expect: {
|
||||
function f(a, b) {
|
||||
return a ?? b;
|
||||
}
|
||||
console.log(f(null), f(null, {}));
|
||||
console.log(f(42), f(42, {}));
|
||||
}
|
||||
expect_stdout: [
|
||||
"undefined {}",
|
||||
"42 42",
|
||||
]
|
||||
node_version: ">=14"
|
||||
}
|
||||
|
||||
de_morgan_2d: {
|
||||
options = {
|
||||
booleans: true,
|
||||
evaluate: true,
|
||||
}
|
||||
input: {
|
||||
function f(a, b) {
|
||||
return a ?? (a && b);
|
||||
}
|
||||
console.log(f(null), f(null, {}));
|
||||
console.log(f(42), f(42, {}));
|
||||
}
|
||||
expect: {
|
||||
function f(a, b) {
|
||||
return a;
|
||||
}
|
||||
console.log(f(null), f(null, {}));
|
||||
console.log(f(42), f(42, {}));
|
||||
}
|
||||
expect_stdout: [
|
||||
"null null",
|
||||
"42 42",
|
||||
]
|
||||
node_version: ">=14"
|
||||
}
|
||||
|
||||
de_morgan_2e: {
|
||||
options = {
|
||||
booleans: true,
|
||||
conditionals: true,
|
||||
}
|
||||
input: {
|
||||
function f(a, b) {
|
||||
return a ?? (a ?? b);
|
||||
}
|
||||
console.log(f(null), f(null, {}));
|
||||
console.log(f(42), f(42, {}));
|
||||
}
|
||||
expect: {
|
||||
function f(a, b) {
|
||||
return a ?? b;
|
||||
}
|
||||
console.log(f(null), f(null, {}));
|
||||
console.log(f(42), f(42, {}));
|
||||
}
|
||||
expect_stdout: [
|
||||
"undefined {}",
|
||||
"42 42",
|
||||
]
|
||||
node_version: ">=14"
|
||||
}
|
||||
|
||||
issue_4679: {
|
||||
options = {
|
||||
comparisons: true,
|
||||
ie8: true,
|
||||
ie: true,
|
||||
}
|
||||
input: {
|
||||
var a;
|
||||
|
||||
@@ -1,3 +1,10 @@
|
||||
literal_infinity: {
|
||||
input: {
|
||||
console.log(2e308, -1e2345);
|
||||
}
|
||||
expect_exact: "console.log(1/0,-(1/0));"
|
||||
}
|
||||
|
||||
parentheses_for_prototype_functions: {
|
||||
beautify = {
|
||||
beautify: true,
|
||||
@@ -97,6 +104,40 @@ parentheses_for_prototype_functions_galio: {
|
||||
expect_stdout: true
|
||||
}
|
||||
|
||||
octal: {
|
||||
beautify = {
|
||||
beautify: true,
|
||||
}
|
||||
input: {
|
||||
(function() {
|
||||
console.log(052);
|
||||
console.log(-052);
|
||||
|
||||
console.log(018);
|
||||
console.log(-018);
|
||||
|
||||
console.log(052.toFixed(0));
|
||||
console.log(-052.toFixed(0));
|
||||
|
||||
console.log(018..toFixed(0));
|
||||
console.log(-018..toFixed(0));
|
||||
})();
|
||||
}
|
||||
expect_exact: [
|
||||
"(function() {",
|
||||
" console.log(42);",
|
||||
" console.log(-42);",
|
||||
" console.log(18);",
|
||||
" console.log(-18);",
|
||||
" console.log(42..toFixed(0));",
|
||||
" console.log(-42..toFixed(0));",
|
||||
" console.log(18..toFixed(0));",
|
||||
" console.log(-18..toFixed(0));",
|
||||
"})();",
|
||||
]
|
||||
expect_stdout: true
|
||||
}
|
||||
|
||||
comparisons: {
|
||||
options = {
|
||||
comparisons: true,
|
||||
|
||||
468
test/compress/optional-chains.js
Normal file
468
test/compress/optional-chains.js
Normal file
@@ -0,0 +1,468 @@
|
||||
call: {
|
||||
input: {
|
||||
console.log?.(undefined?.(console.log("FAIL")));
|
||||
}
|
||||
expect_exact: 'console.log?.((void 0)?.(console.log("FAIL")));'
|
||||
expect_stdout: "undefined"
|
||||
node_version: ">=14"
|
||||
}
|
||||
|
||||
dot: {
|
||||
input: {
|
||||
console?.log((void 0)?.p);
|
||||
}
|
||||
expect_exact: "console?.log((void 0)?.p);"
|
||||
expect_stdout: "undefined"
|
||||
node_version: ">=14"
|
||||
}
|
||||
|
||||
dot_in: {
|
||||
input: {
|
||||
var o = { in: 42 };
|
||||
console.log(o.in, o?.in);
|
||||
}
|
||||
expect_exact: "var o={in:42};console.log(o.in,o?.in);"
|
||||
expect_stdout: "42 42"
|
||||
node_version: ">=14"
|
||||
}
|
||||
|
||||
sub: {
|
||||
input: {
|
||||
console?.["log"](null?.[console.log("FAIL")]);
|
||||
}
|
||||
expect_exact: 'console?.["log"](null?.[console.log("FAIL")]);'
|
||||
expect_stdout: "undefined"
|
||||
node_version: ">=14"
|
||||
}
|
||||
|
||||
ternary_decimal: {
|
||||
input: {
|
||||
null ? .42 : console.log("PASS");
|
||||
}
|
||||
expect_exact: 'null?.42:console.log("PASS");'
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
assign_parentheses_call: {
|
||||
input: {
|
||||
var o = {};
|
||||
((() => o)?.()).p = "PASS";
|
||||
console.log(o.p);
|
||||
}
|
||||
expect_exact: 'var o={};((()=>o)?.()).p="PASS";console.log(o.p);'
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=14"
|
||||
}
|
||||
|
||||
assign_parentheses_dot: {
|
||||
input: {
|
||||
(console?.log).name.p = console.log("PASS");
|
||||
}
|
||||
expect_exact: '(console?.log).name.p=console.log("PASS");'
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=14"
|
||||
}
|
||||
|
||||
assign_no_parentheses: {
|
||||
input: {
|
||||
console[console.log?.("PASS")] = 42;
|
||||
}
|
||||
expect_exact: 'console[console.log?.("PASS")]=42;'
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=14"
|
||||
}
|
||||
|
||||
call_parentheses: {
|
||||
input: {
|
||||
(function(o) {
|
||||
console.log(o.f("FAIL"), (o.f)("FAIL"), (0, o.f)(42));
|
||||
console.log(o?.f("FAIL"), (o?.f)("FAIL"), (0, o?.f)(42));
|
||||
})({
|
||||
a: "PASS",
|
||||
f(b) {
|
||||
return this.a || b;
|
||||
},
|
||||
});
|
||||
}
|
||||
expect_exact: '(function(o){console.log(o.f("FAIL"),o.f("FAIL"),(0,o.f)(42));console.log(o?.f("FAIL"),(o?.f)("FAIL"),(0,o?.f)(42))})({a:"PASS",f(b){return this.a||b}});'
|
||||
expect_stdout: [
|
||||
"PASS PASS 42",
|
||||
"PASS PASS 42",
|
||||
]
|
||||
node_version: ">=14"
|
||||
}
|
||||
|
||||
unary_parentheses: {
|
||||
input: {
|
||||
var o = { p: 41 };
|
||||
(function() {
|
||||
return o;
|
||||
}?.()).p++;
|
||||
console.log(o.p);
|
||||
}
|
||||
expect_exact: "var o={p:41};(function(){return o}?.()).p++;console.log(o.p);"
|
||||
expect_stdout: "42"
|
||||
node_version: ">=14"
|
||||
}
|
||||
|
||||
collapse_vars_1: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
}
|
||||
input: {
|
||||
var a;
|
||||
A = 42;
|
||||
a?.[42];
|
||||
console.log(typeof A);
|
||||
}
|
||||
expect: {
|
||||
var a;
|
||||
A = 42;
|
||||
a?.[42];
|
||||
console.log(typeof A);
|
||||
}
|
||||
expect_stdout: "number"
|
||||
node_version: ">=14"
|
||||
}
|
||||
|
||||
collapse_vars_2: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
}
|
||||
input: {
|
||||
var a;
|
||||
A = 42;
|
||||
a?.(42);
|
||||
console.log(typeof A);
|
||||
}
|
||||
expect: {
|
||||
var a;
|
||||
A = 42;
|
||||
a?.(42);
|
||||
console.log(typeof A);
|
||||
}
|
||||
expect_stdout: "number"
|
||||
node_version: ">=14"
|
||||
}
|
||||
|
||||
properties: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
properties: true,
|
||||
}
|
||||
input: {
|
||||
var a;
|
||||
console.log(a?.["FAIL"]);
|
||||
}
|
||||
expect: {
|
||||
var a;
|
||||
console.log(a?.FAIL);
|
||||
}
|
||||
expect_stdout: "undefined"
|
||||
node_version: ">=14"
|
||||
}
|
||||
|
||||
reduce_vars_1: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
reduce_vars: true,
|
||||
toplevel: true,
|
||||
}
|
||||
input: {
|
||||
var a = 1;
|
||||
null?.[a = 0];
|
||||
console.log(a ? "PASS" : "FAIL");
|
||||
}
|
||||
expect: {
|
||||
var a = 1;
|
||||
null?.[a = 0];
|
||||
console.log(a ? "PASS" : "FAIL");
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=14"
|
||||
}
|
||||
|
||||
reduce_vars_2: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
reduce_vars: true,
|
||||
toplevel: true,
|
||||
}
|
||||
input: {
|
||||
var a = 1;
|
||||
null?.(a = 0);
|
||||
console.log(a ? "PASS" : "FAIL");
|
||||
}
|
||||
expect: {
|
||||
var a = 1;
|
||||
null?.(a = 0);
|
||||
console.log(a ? "PASS" : "FAIL");
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=14"
|
||||
}
|
||||
|
||||
side_effects: {
|
||||
options = {
|
||||
side_effects: true,
|
||||
}
|
||||
input: {
|
||||
var a;
|
||||
a?.[a = "FAIL"];
|
||||
console.log(a);
|
||||
}
|
||||
expect: {
|
||||
var a;
|
||||
a?.[a = "FAIL"];
|
||||
console.log(a);
|
||||
}
|
||||
expect_stdout: "undefined"
|
||||
node_version: ">=14"
|
||||
}
|
||||
|
||||
trim_1: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
optional_chains: true,
|
||||
reduce_vars: true,
|
||||
unsafe: true,
|
||||
}
|
||||
input: {
|
||||
(function(a, b) {
|
||||
console?.log?.(a?.p, b?.[console.log("FAIL")]);
|
||||
})?.({ p: "PASS" });
|
||||
}
|
||||
expect: {
|
||||
(function(a, b) {
|
||||
console?.log?.(a.p, void 0);
|
||||
})({ p: "PASS" });
|
||||
}
|
||||
expect_stdout: "PASS undefined"
|
||||
node_version: ">=14"
|
||||
}
|
||||
|
||||
trim_2: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
optional_chains: true,
|
||||
side_effects: true,
|
||||
}
|
||||
input: {
|
||||
(void console.log("PASS"))?.[console.log("FAIL")];
|
||||
}
|
||||
expect: {
|
||||
console.log("PASS");
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=14"
|
||||
}
|
||||
|
||||
trim_dot_call_1: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
optional_chains: true,
|
||||
}
|
||||
input: {
|
||||
console.log(null?.f());
|
||||
}
|
||||
expect: {
|
||||
console.log(void 0);
|
||||
}
|
||||
expect_stdout: "undefined"
|
||||
node_version: ">=14"
|
||||
}
|
||||
|
||||
trim_dot_call_2: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
optional_chains: true,
|
||||
unsafe: true,
|
||||
}
|
||||
input: {
|
||||
try {
|
||||
(null?.p)();
|
||||
} catch (e) {
|
||||
console.log("PASS");
|
||||
}
|
||||
}
|
||||
expect: {
|
||||
try {
|
||||
(void 0)();
|
||||
} catch (e) {
|
||||
console.log("PASS");
|
||||
}
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=14"
|
||||
}
|
||||
|
||||
trim_dot_call_3: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
optional_chains: true,
|
||||
unsafe: true,
|
||||
}
|
||||
input: {
|
||||
try {
|
||||
({ p: null })?.p();
|
||||
} catch (e) {
|
||||
console.log("PASS");
|
||||
}
|
||||
}
|
||||
expect: {
|
||||
try {
|
||||
null();
|
||||
} catch (e) {
|
||||
console.log("PASS");
|
||||
}
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=14"
|
||||
}
|
||||
|
||||
trim_dot_sub: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
optional_chains: true,
|
||||
}
|
||||
input: {
|
||||
console.log(null?.p[42]);
|
||||
}
|
||||
expect: {
|
||||
console.log(void 0);
|
||||
}
|
||||
expect_stdout: "undefined"
|
||||
node_version: ">=14"
|
||||
}
|
||||
|
||||
trim_sub_call_call: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
optional_chains: true,
|
||||
}
|
||||
input: {
|
||||
console.log(null?.[42]()());
|
||||
}
|
||||
expect: {
|
||||
console.log(void 0);
|
||||
}
|
||||
expect_stdout: "undefined"
|
||||
node_version: ">=14"
|
||||
}
|
||||
|
||||
issue_4906: {
|
||||
options = {
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
do {
|
||||
var a = a?.[42];
|
||||
} while (console.log("PASS"));
|
||||
}
|
||||
expect: {
|
||||
do {} while (console.log("PASS"));
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=14"
|
||||
}
|
||||
|
||||
issue_4928: {
|
||||
options = {
|
||||
ie: true,
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var a = a?.[function f() {
|
||||
f(a);
|
||||
}];
|
||||
console.log(typeof f);
|
||||
}
|
||||
expect: {
|
||||
var a = a?.[function f() {
|
||||
f(a);
|
||||
}];
|
||||
console.log(typeof f);
|
||||
}
|
||||
expect_stdout: "undefined"
|
||||
node_version: ">=14"
|
||||
}
|
||||
|
||||
issue_4947_1: {
|
||||
options = {
|
||||
conditionals: true,
|
||||
}
|
||||
input: {
|
||||
console.log(console.foo ? 42..p : console.bar?.p);
|
||||
}
|
||||
expect: {
|
||||
console.log(console.foo ? 42..p : console.bar?.p);
|
||||
}
|
||||
expect_stdout: "undefined"
|
||||
node_version: ">=14"
|
||||
}
|
||||
|
||||
issue_4947_2: {
|
||||
options = {
|
||||
conditionals: true,
|
||||
}
|
||||
input: {
|
||||
var log = console.log, fail;
|
||||
log("PASS") ? log(42) : fail?.(42);
|
||||
}
|
||||
expect: {
|
||||
var log = console.log, fail;
|
||||
log("PASS") ? log(42) : fail?.(42);
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=14"
|
||||
}
|
||||
|
||||
issue_5039: {
|
||||
options = {
|
||||
ie: true,
|
||||
side_effects: true,
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var a = a?.[function f() {
|
||||
f;
|
||||
a;
|
||||
}];
|
||||
console.log("PASS");
|
||||
}
|
||||
expect: {
|
||||
(function f() {});
|
||||
console.log("PASS");
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=14"
|
||||
}
|
||||
|
||||
issue_5091: {
|
||||
options = {
|
||||
merge_vars: true,
|
||||
}
|
||||
input: {
|
||||
function f(a) {
|
||||
var b = a.p;
|
||||
var c;
|
||||
b?.[c = "FAIL 2"];
|
||||
return b || c;
|
||||
}
|
||||
console.log(f("FAIL 1") || "PASS");
|
||||
}
|
||||
expect: {
|
||||
function f(b) {
|
||||
var b = b.p;
|
||||
var c;
|
||||
b?.[c = "FAIL 2"];
|
||||
return b || c;
|
||||
}
|
||||
console.log(f("FAIL 1") || "PASS");
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=14"
|
||||
}
|
||||
@@ -17,7 +17,7 @@ dot_properties: {
|
||||
properties: true,
|
||||
}
|
||||
beautify = {
|
||||
ie8: true,
|
||||
ie: true,
|
||||
}
|
||||
input: {
|
||||
a["foo"] = "bar";
|
||||
@@ -43,7 +43,7 @@ dot_properties_es5: {
|
||||
properties: true,
|
||||
}
|
||||
beautify = {
|
||||
ie8: false,
|
||||
ie: false,
|
||||
}
|
||||
input: {
|
||||
a["foo"] = "bar";
|
||||
@@ -1400,3 +1400,113 @@ object_super: {
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
issue_4831_1: {
|
||||
options = {
|
||||
properties: true,
|
||||
}
|
||||
input: {
|
||||
console.log({
|
||||
f() {
|
||||
return arguments;
|
||||
},
|
||||
}.f("PASS")[0]);
|
||||
}
|
||||
expect: {
|
||||
console.log([
|
||||
function() {
|
||||
return arguments;
|
||||
},
|
||||
][0]("PASS")[0]);
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
issue_4831_2: {
|
||||
options = {
|
||||
properties: true,
|
||||
}
|
||||
input: {
|
||||
var f = {
|
||||
f() {
|
||||
return arguments;
|
||||
},
|
||||
}.f;
|
||||
console.log(f("PASS")[0]);
|
||||
}
|
||||
expect: {
|
||||
var f = {
|
||||
f() {
|
||||
return arguments;
|
||||
},
|
||||
}.f;
|
||||
console.log(f("PASS")[0]);
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
issue_4888: {
|
||||
options = {
|
||||
properties: true,
|
||||
}
|
||||
input: {
|
||||
console.log(typeof {
|
||||
__proto__: 42,
|
||||
}.__proto__);
|
||||
}
|
||||
expect: {
|
||||
console.log(typeof {
|
||||
__proto__: 42,
|
||||
}.__proto__);
|
||||
}
|
||||
expect_stdout: "object"
|
||||
}
|
||||
|
||||
issue_5093: {
|
||||
beautify = {
|
||||
keep_quoted_props: true,
|
||||
}
|
||||
input: {
|
||||
console.log({
|
||||
a: true,
|
||||
'42': "PASS",
|
||||
"null": [],
|
||||
}[6 * 7]);
|
||||
}
|
||||
expect_exact: 'console.log({a:true,"42":"PASS","null":[]}[6*7]);'
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
issue_5093_quote_keys: {
|
||||
beautify = {
|
||||
keep_quoted_props: true,
|
||||
quote_keys: true,
|
||||
}
|
||||
input: {
|
||||
console.log({
|
||||
a: true,
|
||||
'42': "PASS",
|
||||
"null": [],
|
||||
}[6 * 7]);
|
||||
}
|
||||
expect_exact: 'console.log({"a":true,"42":"PASS","null":[]}[6*7]);'
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
issue_5093_quote_style: {
|
||||
beautify = {
|
||||
keep_quoted_props: true,
|
||||
quote_style: 3,
|
||||
}
|
||||
input: {
|
||||
console.log({
|
||||
a: true,
|
||||
'42': "PASS",
|
||||
"null": [],
|
||||
}[6 * 7]);
|
||||
}
|
||||
expect_exact: 'console.log({a:true,\'42\':"PASS","null":[]}[6*7]);'
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
@@ -133,7 +133,7 @@ conditional: {
|
||||
relational: {
|
||||
options = {
|
||||
pure_funcs: [ "foo" ],
|
||||
side_effects :true,
|
||||
side_effects: true,
|
||||
}
|
||||
input: {
|
||||
foo() in new foo();
|
||||
@@ -158,7 +158,7 @@ relational: {
|
||||
arithmetic: {
|
||||
options = {
|
||||
pure_funcs: [ "foo" ],
|
||||
side_effects :true,
|
||||
side_effects: true,
|
||||
}
|
||||
input: {
|
||||
foo() + foo();
|
||||
@@ -183,7 +183,7 @@ arithmetic: {
|
||||
boolean_and: {
|
||||
options = {
|
||||
pure_funcs: [ "foo" ],
|
||||
side_effects :true,
|
||||
side_effects: true,
|
||||
}
|
||||
input: {
|
||||
foo() && foo();
|
||||
@@ -208,7 +208,7 @@ boolean_and: {
|
||||
boolean_or: {
|
||||
options = {
|
||||
pure_funcs: [ "foo" ],
|
||||
side_effects :true,
|
||||
side_effects: true,
|
||||
}
|
||||
input: {
|
||||
foo() || foo();
|
||||
@@ -233,7 +233,7 @@ boolean_or: {
|
||||
assign: {
|
||||
options = {
|
||||
pure_funcs: [ "foo" ],
|
||||
side_effects :true,
|
||||
side_effects: true,
|
||||
}
|
||||
input: {
|
||||
var a;
|
||||
@@ -256,7 +256,7 @@ assign: {
|
||||
unary: {
|
||||
options = {
|
||||
pure_funcs: [ "foo" ],
|
||||
side_effects :true,
|
||||
side_effects: true,
|
||||
}
|
||||
input: {
|
||||
typeof foo();
|
||||
@@ -294,248 +294,6 @@ unary: {
|
||||
}
|
||||
}
|
||||
|
||||
issue_2629_1: {
|
||||
options = {
|
||||
side_effects: true,
|
||||
}
|
||||
beautify = {
|
||||
comments: "all",
|
||||
}
|
||||
input: {
|
||||
/*@__PURE__*/ a();
|
||||
/*@__PURE__*/ (b());
|
||||
(/*@__PURE__*/ c)();
|
||||
(/*@__PURE__*/ d());
|
||||
}
|
||||
expect_exact: [
|
||||
"/* */c();",
|
||||
]
|
||||
}
|
||||
|
||||
issue_2629_2: {
|
||||
options = {
|
||||
side_effects: true,
|
||||
}
|
||||
beautify = {
|
||||
comments: "all",
|
||||
}
|
||||
input: {
|
||||
/*@__PURE__*/ a(1)(2)(3);
|
||||
/*@__PURE__*/ (b(1))(2)(3);
|
||||
/*@__PURE__*/ (c(1)(2))(3);
|
||||
/*@__PURE__*/ (d(1)(2)(3));
|
||||
(/*@__PURE__*/ e)(1)(2)(3);
|
||||
(/*@__PURE__*/ f(1))(2)(3);
|
||||
(/*@__PURE__*/ g(1)(2))(3);
|
||||
(/*@__PURE__*/ h(1)(2)(3));
|
||||
}
|
||||
expect_exact: [
|
||||
"/* */e(1)(2)(3);",
|
||||
"/* */f(1)(2)(3);",
|
||||
"/* */g(1)(2)(3);",
|
||||
]
|
||||
}
|
||||
|
||||
issue_2629_3: {
|
||||
options = {
|
||||
side_effects: true,
|
||||
}
|
||||
beautify = {
|
||||
comments: "all",
|
||||
}
|
||||
input: {
|
||||
/*@__PURE__*/ a.x(1).y(2).z(3);
|
||||
/*@__PURE__*/ (b.x)(1).y(2).z(3);
|
||||
/*@__PURE__*/ (c.x(1)).y(2).z(3);
|
||||
/*@__PURE__*/ (d.x(1).y)(2).z(3);
|
||||
/*@__PURE__*/ (e.x(1).y(2)).z(3);
|
||||
/*@__PURE__*/ (f.x(1).y(2).z)(3);
|
||||
/*@__PURE__*/ (g.x(1).y(2).z(3));
|
||||
(/*@__PURE__*/ h).x(1).y(2).z(3);
|
||||
(/*@__PURE__*/ i.x)(1).y(2).z(3);
|
||||
(/*@__PURE__*/ j.x(1)).y(2).z(3);
|
||||
(/*@__PURE__*/ k.x(1).y)(2).z(3);
|
||||
(/*@__PURE__*/ l.x(1).y(2)).z(3);
|
||||
(/*@__PURE__*/ m.x(1).y(2).z)(3);
|
||||
(/*@__PURE__*/ n.x(1).y(2).z(3));
|
||||
}
|
||||
expect_exact: [
|
||||
"/* */h.x(1).y(2).z(3);",
|
||||
"/* */i.x(1).y(2).z(3);",
|
||||
"/* */j.x(1).y(2).z(3);",
|
||||
"/* */k.x(1).y(2).z(3);",
|
||||
"/* */l.x(1).y(2).z(3);",
|
||||
"/* */m.x(1).y(2).z(3);",
|
||||
]
|
||||
}
|
||||
|
||||
issue_2629_4: {
|
||||
options = {
|
||||
side_effects: true,
|
||||
}
|
||||
input: {
|
||||
(/*@__PURE__*/ x(), y());
|
||||
(w(), /*@__PURE__*/ x(), y());
|
||||
}
|
||||
expect: {
|
||||
y();
|
||||
w(), y();
|
||||
}
|
||||
}
|
||||
|
||||
issue_2629_5: {
|
||||
options = {
|
||||
side_effects: true,
|
||||
}
|
||||
input: {
|
||||
[ /*@__PURE__*/ x() ];
|
||||
[ /*@__PURE__*/ x(), y() ];
|
||||
[ w(), /*@__PURE__*/ x(), y() ];
|
||||
}
|
||||
expect: {
|
||||
y();
|
||||
w(), y();
|
||||
}
|
||||
}
|
||||
|
||||
issue_2638: {
|
||||
options = {
|
||||
side_effects: true,
|
||||
}
|
||||
beautify = {
|
||||
comments: "all",
|
||||
}
|
||||
input: {
|
||||
/*@__PURE__*/(g() || h())(x(), y());
|
||||
(/*@__PURE__*/ (a() || b()))(c(), d());
|
||||
}
|
||||
expect_exact: [
|
||||
"/* */x(),y();",
|
||||
"/* */(a()||b())(c(),d());",
|
||||
]
|
||||
}
|
||||
|
||||
issue_2705_1: {
|
||||
options = {
|
||||
side_effects: true,
|
||||
}
|
||||
beautify = {
|
||||
comments: "all",
|
||||
}
|
||||
input: {
|
||||
/*@__PURE__*/ new a();
|
||||
/*@__PURE__*/ (new b());
|
||||
new (/*@__PURE__*/ c)();
|
||||
(/*@__PURE__*/ new d());
|
||||
}
|
||||
expect_exact: [
|
||||
"new/* */c;",
|
||||
]
|
||||
}
|
||||
|
||||
issue_2705_2: {
|
||||
options = {
|
||||
side_effects: true,
|
||||
}
|
||||
beautify = {
|
||||
comments: "all",
|
||||
}
|
||||
input: {
|
||||
/*@__PURE__*/ new a(1)(2)(3);
|
||||
/*@__PURE__*/ new (b(1))(2)(3);
|
||||
/*@__PURE__*/ new (c(1)(2))(3);
|
||||
/*@__PURE__*/ new (d(1)(2)(3));
|
||||
new (/*@__PURE__*/ e)(1)(2)(3);
|
||||
(/*@__PURE__*/ new f(1))(2)(3);
|
||||
(/*@__PURE__*/ new g(1)(2))(3);
|
||||
(/*@__PURE__*/ new h(1)(2)(3));
|
||||
}
|
||||
expect_exact: [
|
||||
"new/* */e(1)(2)(3);",
|
||||
"/* */new f(1)(2)(3);",
|
||||
"/* */new g(1)(2)(3);",
|
||||
]
|
||||
}
|
||||
|
||||
issue_2705_3: {
|
||||
options = {
|
||||
side_effects: true,
|
||||
}
|
||||
beautify = {
|
||||
comments: "all",
|
||||
}
|
||||
input: {
|
||||
/*@__PURE__*/ new a.x(1).y(2).z(3);
|
||||
/*@__PURE__*/ new (b.x)(1).y(2).z(3);
|
||||
/*@__PURE__*/ new (c.x(1)).y(2).z(3);
|
||||
/*@__PURE__*/ new (d.x(1).y)(2).z(3);
|
||||
/*@__PURE__*/ new (e.x(1).y(2)).z(3);
|
||||
/*@__PURE__*/ new (f.x(1).y(2).z)(3);
|
||||
/*@__PURE__*/ new (g.x(1).y(2).z(3));
|
||||
new (/*@__PURE__*/ h).x(1).y(2).z(3);
|
||||
/* */ new (/*@__PURE__*/ i.x)(1).y(2).z(3);
|
||||
(/*@__PURE__*/ new j.x(1)).y(2).z(3);
|
||||
(/*@__PURE__*/ new k.x(1).y)(2).z(3);
|
||||
(/*@__PURE__*/ new l.x(1).y(2)).z(3);
|
||||
(/*@__PURE__*/ new m.x(1).y(2).z)(3);
|
||||
(/*@__PURE__*/ new n.x(1).y(2).z(3));
|
||||
}
|
||||
expect_exact: [
|
||||
"new/* */h.x(1).y(2).z(3);",
|
||||
"/* */new/* */i.x(1).y(2).z(3);",
|
||||
"/* */new j.x(1).y(2).z(3);",
|
||||
"/* */new k.x(1).y(2).z(3);",
|
||||
"/* */new l.x(1).y(2).z(3);",
|
||||
"/* */new m.x(1).y(2).z(3);",
|
||||
]
|
||||
}
|
||||
|
||||
issue_2705_4: {
|
||||
options = {
|
||||
side_effects: true,
|
||||
}
|
||||
input: {
|
||||
(/*@__PURE__*/ new x(), y());
|
||||
(w(), /*@__PURE__*/ new x(), y());
|
||||
}
|
||||
expect: {
|
||||
y();
|
||||
w(), y();
|
||||
}
|
||||
}
|
||||
|
||||
issue_2705_5: {
|
||||
options = {
|
||||
side_effects: true,
|
||||
}
|
||||
input: {
|
||||
[ /*@__PURE__*/ new x() ];
|
||||
[ /*@__PURE__*/ new x(), y() ];
|
||||
[ w(), /*@__PURE__*/ new x(), y() ];
|
||||
}
|
||||
expect: {
|
||||
y();
|
||||
w(), y();
|
||||
}
|
||||
}
|
||||
|
||||
issue_2705_6: {
|
||||
options = {
|
||||
side_effects: true,
|
||||
}
|
||||
beautify = {
|
||||
comments: "all",
|
||||
}
|
||||
input: {
|
||||
/*@__PURE__*/new (g() || h())(x(), y());
|
||||
/* */ new (/*@__PURE__*/ (a() || b()))(c(), d());
|
||||
}
|
||||
expect_exact: [
|
||||
"/* */x(),y();",
|
||||
"/* */new(/* */a()||b())(c(),d());",
|
||||
]
|
||||
}
|
||||
|
||||
issue_3065_1: {
|
||||
options = {
|
||||
inline: true,
|
||||
@@ -680,130 +438,3 @@ issue_3325_2: {
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
issue_3858: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
inline: true,
|
||||
keep_fargs: false,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var f = function(a) {
|
||||
return /*@__PURE__*/ function(b) {
|
||||
console.log(b);
|
||||
}(a);
|
||||
};
|
||||
f("PASS");
|
||||
}
|
||||
expect: {
|
||||
var f = function(a) {
|
||||
return function() {
|
||||
console.log(a);
|
||||
}();
|
||||
};
|
||||
f("PASS");
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
inline_pure_call_1: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
inline: true,
|
||||
keep_fargs: false,
|
||||
reduce_vars: true,
|
||||
sequences: true,
|
||||
side_effects: true,
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var f = function(a) {
|
||||
return /*@__PURE__*/ function(b) {
|
||||
console.log(b);
|
||||
}(a);
|
||||
};
|
||||
f("PASS");
|
||||
}
|
||||
expect: {}
|
||||
}
|
||||
|
||||
inline_pure_call_2: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
inline: true,
|
||||
keep_fargs: false,
|
||||
reduce_vars: true,
|
||||
sequences: true,
|
||||
side_effects: true,
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var f = function(a) {
|
||||
return /*@__PURE__*/ function(b) {
|
||||
console.log(b);
|
||||
}(a);
|
||||
};
|
||||
var a = f("PASS");
|
||||
}
|
||||
expect: {}
|
||||
}
|
||||
|
||||
inline_pure_call_3: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
evaluate: true,
|
||||
inline: true,
|
||||
keep_fargs: false,
|
||||
passes: 2,
|
||||
reduce_vars: true,
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var f = function(a) {
|
||||
return /*@__PURE__*/ function(b) {
|
||||
console.log(b);
|
||||
}(a);
|
||||
};
|
||||
var a = f("PASS");
|
||||
console.log(a);
|
||||
}
|
||||
expect: {
|
||||
var a = function() {
|
||||
console.log("PASS");
|
||||
}();
|
||||
console.log(a);
|
||||
}
|
||||
expect_stdout: [
|
||||
"PASS",
|
||||
"undefined",
|
||||
]
|
||||
}
|
||||
|
||||
inline_pure_call_4: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
reduce_vars: true,
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var a = /*@__PURE__*/ function() {
|
||||
return console.log("PASS"), 42;
|
||||
}();
|
||||
console.log(a);
|
||||
}
|
||||
expect: {
|
||||
var a = function() {
|
||||
return console.log("PASS"), 42;
|
||||
}();
|
||||
console.log(a);
|
||||
}
|
||||
expect_stdout: [
|
||||
"PASS",
|
||||
"42",
|
||||
]
|
||||
}
|
||||
|
||||
@@ -1321,10 +1321,10 @@ issue_2878: {
|
||||
issue_3427: {
|
||||
options = {
|
||||
assignments: true,
|
||||
collapse_vars: true,
|
||||
evaluate: true,
|
||||
inline: true,
|
||||
passes: 2,
|
||||
pure_getters: "strict",
|
||||
reduce_vars: true,
|
||||
sequences: true,
|
||||
side_effects: true,
|
||||
toplevel: true,
|
||||
@@ -1490,3 +1490,178 @@ issue_4751: {
|
||||
};
|
||||
}
|
||||
}
|
||||
|
||||
super_toString: {
|
||||
options = {
|
||||
pure_getters: true,
|
||||
unsafe: true,
|
||||
}
|
||||
input: {
|
||||
console.log({
|
||||
f() {
|
||||
return super.toString();
|
||||
},
|
||||
}.f());
|
||||
}
|
||||
expect: {
|
||||
console.log({
|
||||
f() {
|
||||
return super.toString();
|
||||
},
|
||||
}.f());
|
||||
}
|
||||
expect_stdout: "[object Object]"
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
this_toString: {
|
||||
options = {
|
||||
pure_getters: true,
|
||||
unsafe: true,
|
||||
}
|
||||
input: {
|
||||
console.log({
|
||||
f() {
|
||||
return this.toString();
|
||||
},
|
||||
}.f());
|
||||
}
|
||||
expect: {
|
||||
console.log({
|
||||
f() {
|
||||
return "" + this;
|
||||
},
|
||||
}.f());
|
||||
}
|
||||
expect_stdout: "[object Object]"
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
issue_4803: {
|
||||
options = {
|
||||
hoist_vars: true,
|
||||
pure_getters: "strict",
|
||||
reduce_vars: true,
|
||||
side_effects: true,
|
||||
toplevel: true,
|
||||
}
|
||||
input: {
|
||||
var o = {
|
||||
get f() {
|
||||
console.log("PASS");
|
||||
},
|
||||
} || 42;
|
||||
for (var k in o)
|
||||
o[k];
|
||||
}
|
||||
expect: {
|
||||
var k, o = {
|
||||
get f() {
|
||||
console.log("PASS");
|
||||
},
|
||||
} || 42;
|
||||
for (k in o)
|
||||
o[k];
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
nested_property_assignments_1: {
|
||||
options = {
|
||||
pure_getters: "strict",
|
||||
reduce_vars: true,
|
||||
sequences: true,
|
||||
side_effects: true,
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var f;
|
||||
((f = function() {
|
||||
console.log("FAIL");
|
||||
}).p = f).q = console.log("PASS");
|
||||
}
|
||||
expect: {
|
||||
console.log("PASS");
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
nested_property_assignments_2: {
|
||||
options = {
|
||||
pure_getters: "strict",
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var o = {};
|
||||
(function() {
|
||||
var a;
|
||||
(o.p = a = {}).q = "PASS";
|
||||
})();
|
||||
console.log(o.p.q);
|
||||
}
|
||||
expect: {
|
||||
var o = {};
|
||||
(function() {
|
||||
(o.p = {}).q = "PASS";
|
||||
})();
|
||||
console.log(o.p.q);
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
nested_property_assignments_3: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
pure_getters: true,
|
||||
side_effects: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var o = { p: {} };
|
||||
(function(a) {
|
||||
console && a;
|
||||
if (console) {
|
||||
a = a.p;
|
||||
a.q = a;
|
||||
}
|
||||
})(o);
|
||||
console.log(o.p.q === o.p ? "PASS" : "FAIL");
|
||||
}
|
||||
expect: {
|
||||
var o = { p: {} };
|
||||
(function(a) {
|
||||
console;
|
||||
if (console)
|
||||
(a = a.p).q = a;
|
||||
})(o);
|
||||
console.log(o.p.q === o.p ? "PASS" : "FAIL");
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
issue_4939: {
|
||||
options = {
|
||||
pure_getters: "strict",
|
||||
side_effects: true,
|
||||
}
|
||||
input: {
|
||||
({
|
||||
__proto__: {
|
||||
get p() {
|
||||
console.log("PASS");
|
||||
},
|
||||
},
|
||||
}).p;
|
||||
}
|
||||
expect: {
|
||||
({
|
||||
__proto__: {
|
||||
get p() {
|
||||
console.log("PASS");
|
||||
},
|
||||
},
|
||||
}).p;
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
@@ -1292,6 +1292,7 @@ toplevel_on_loops_3: {
|
||||
loops: true,
|
||||
reduce_funcs: true,
|
||||
reduce_vars: true,
|
||||
side_effects: true,
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
}
|
||||
@@ -2071,72 +2072,6 @@ issue_1670_2: {
|
||||
}
|
||||
|
||||
issue_1670_3: {
|
||||
options = {
|
||||
comparisons: true,
|
||||
conditionals: true,
|
||||
dead_code: true,
|
||||
evaluate: true,
|
||||
reduce_funcs: true,
|
||||
reduce_vars: true,
|
||||
side_effects: true,
|
||||
switches: true,
|
||||
typeofs: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
(function f() {
|
||||
switch (1) {
|
||||
case 0:
|
||||
var a = true;
|
||||
break;
|
||||
case 1:
|
||||
if (typeof a === "undefined") console.log("PASS");
|
||||
else console.log("FAIL");
|
||||
}
|
||||
})();
|
||||
}
|
||||
expect: {
|
||||
(function() {
|
||||
var a;
|
||||
void 0 === a ? console.log("PASS") : console.log("FAIL");
|
||||
})();
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
issue_1670_4: {
|
||||
options = {
|
||||
conditionals: true,
|
||||
dead_code: true,
|
||||
evaluate: true,
|
||||
passes: 2,
|
||||
reduce_funcs: true,
|
||||
reduce_vars: true,
|
||||
side_effects: true,
|
||||
switches: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
(function f() {
|
||||
switch (1) {
|
||||
case 0:
|
||||
var a = true;
|
||||
break;
|
||||
case 1:
|
||||
if (typeof a === "undefined") console.log("PASS");
|
||||
else console.log("FAIL");
|
||||
}
|
||||
})();
|
||||
}
|
||||
expect: {
|
||||
(function() {
|
||||
console.log("PASS");
|
||||
})();
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
issue_1670_5: {
|
||||
options = {
|
||||
conditionals: true,
|
||||
dead_code: true,
|
||||
@@ -2168,7 +2103,7 @@ issue_1670_5: {
|
||||
expect_stdout: "1"
|
||||
}
|
||||
|
||||
issue_1670_6: {
|
||||
issue_1670_4: {
|
||||
options = {
|
||||
conditionals: true,
|
||||
dead_code: true,
|
||||
@@ -3158,7 +3093,7 @@ accessor_1: {
|
||||
a = 2;
|
||||
return a;
|
||||
},
|
||||
b: 1
|
||||
b: 1,
|
||||
}.b, a);
|
||||
}
|
||||
expect: {
|
||||
@@ -3168,7 +3103,7 @@ accessor_1: {
|
||||
a = 2;
|
||||
return a;
|
||||
},
|
||||
b: 1
|
||||
b: 1,
|
||||
}.b, a);
|
||||
}
|
||||
expect_stdout: "1 1"
|
||||
@@ -3188,7 +3123,7 @@ accessor_2: {
|
||||
var B = {
|
||||
get c() {
|
||||
console.log(A);
|
||||
}
|
||||
},
|
||||
};
|
||||
B.c;
|
||||
}
|
||||
@@ -3196,7 +3131,7 @@ accessor_2: {
|
||||
({
|
||||
get c() {
|
||||
console.log(1);
|
||||
}
|
||||
},
|
||||
}).c;
|
||||
}
|
||||
expect_stdout: "1"
|
||||
@@ -3242,7 +3177,7 @@ obj_var_1: {
|
||||
var obj = {
|
||||
bar: function() {
|
||||
return C + C;
|
||||
}
|
||||
},
|
||||
};
|
||||
console.log(obj.bar());
|
||||
}
|
||||
@@ -3250,7 +3185,7 @@ obj_var_1: {
|
||||
console.log({
|
||||
bar: function() {
|
||||
return 2;
|
||||
}
|
||||
},
|
||||
}.bar());
|
||||
}
|
||||
expect_stdout: "2"
|
||||
@@ -3274,7 +3209,7 @@ obj_var_2: {
|
||||
var obj = {
|
||||
bar: function() {
|
||||
return C + C;
|
||||
}
|
||||
},
|
||||
};
|
||||
console.log(obj.bar());
|
||||
}
|
||||
@@ -4488,6 +4423,7 @@ perf_2: {
|
||||
|
||||
perf_3: {
|
||||
options = {
|
||||
passes: 2,
|
||||
reduce_funcs: true,
|
||||
reduce_vars: true,
|
||||
toplevel: true,
|
||||
@@ -4496,10 +4432,10 @@ perf_3: {
|
||||
input: {
|
||||
var foo = function(x, y, z) {
|
||||
return x < y ? x * y + z : x * z - y;
|
||||
}
|
||||
};
|
||||
var indirect_foo = function(x, y, z) {
|
||||
return foo(x, y, z);
|
||||
}
|
||||
};
|
||||
var sum = 0;
|
||||
for (var i = 0; i < 100; ++i)
|
||||
sum += indirect_foo(i, i + 1, 3 * i);
|
||||
@@ -4528,10 +4464,10 @@ perf_4: {
|
||||
input: {
|
||||
var foo = function(x, y, z) {
|
||||
return x < y ? x * y + z : x * z - y;
|
||||
}
|
||||
};
|
||||
var indirect_foo = function(x, y, z) {
|
||||
return foo(x, y, z);
|
||||
}
|
||||
};
|
||||
var sum = 0;
|
||||
for (var i = 0; i < 100; ++i)
|
||||
sum += indirect_foo(i, i + 1, 3 * i);
|
||||
@@ -4540,10 +4476,10 @@ perf_4: {
|
||||
expect: {
|
||||
var foo = function(x, y, z) {
|
||||
return x < y ? x * y + z : x * z - y;
|
||||
}
|
||||
};
|
||||
var indirect_foo = function(x, y, z) {
|
||||
return foo(x, y, z);
|
||||
}
|
||||
};
|
||||
var sum = 0;
|
||||
for (var i = 0; i < 100; ++i)
|
||||
sum += indirect_foo(i, i + 1, 3 * i);
|
||||
@@ -4632,9 +4568,9 @@ perf_7: {
|
||||
var indirect_foo = function(x, y, z) {
|
||||
var foo = function(x, y, z) {
|
||||
return x < y ? x * y + z : x * z - y;
|
||||
}
|
||||
};
|
||||
return foo(x, y, z);
|
||||
}
|
||||
};
|
||||
var sum = 0;
|
||||
for (var i = 0; i < 100; ++i)
|
||||
sum += indirect_foo(i, i + 1, 3 * i);
|
||||
@@ -4664,9 +4600,9 @@ perf_8: {
|
||||
var indirect_foo = function(x, y, z) {
|
||||
var foo = function(x, y, z) {
|
||||
return x < y ? x * y + z : x * z - y;
|
||||
}
|
||||
};
|
||||
return foo(x, y, z);
|
||||
}
|
||||
};
|
||||
var sum = 0;
|
||||
for (var i = 0; i < 100; ++i)
|
||||
sum += indirect_foo(i, i + 1, 3 * i);
|
||||
@@ -4677,7 +4613,7 @@ perf_8: {
|
||||
return function(x, y, z) {
|
||||
return x < y ? x * y + z : x * z - y;
|
||||
}(x, y, z);
|
||||
}
|
||||
};
|
||||
var sum = 0;
|
||||
for (var i = 0; i < 100; ++i)
|
||||
sum += indirect_foo(i, i + 1, 3 * i);
|
||||
@@ -7631,3 +7567,208 @@ issue_4568: {
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
issue_4937: {
|
||||
options = {
|
||||
reduce_vars: true,
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
function f() {
|
||||
while (console.log("PASS"));
|
||||
}
|
||||
do {
|
||||
function g() {
|
||||
f();
|
||||
}
|
||||
} while (!g);
|
||||
f();
|
||||
}
|
||||
expect: {
|
||||
function f() {
|
||||
while (console.log("PASS"));
|
||||
}
|
||||
do {
|
||||
function g() {
|
||||
f();
|
||||
}
|
||||
} while (!g);
|
||||
f();
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
issue_4943_1: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
reduce_vars: true,
|
||||
toplevel: true,
|
||||
}
|
||||
input: {
|
||||
var a, b = 1;
|
||||
(function f() {
|
||||
a = "foo";
|
||||
b-- && f();
|
||||
console.log(a);
|
||||
a = "bar";
|
||||
})();
|
||||
}
|
||||
expect: {
|
||||
var a, b = 1;
|
||||
(function f() {
|
||||
a = "foo";
|
||||
b-- && f();
|
||||
console.log(a);
|
||||
a = "bar";
|
||||
})();
|
||||
}
|
||||
expect_stdout: [
|
||||
"foo",
|
||||
"bar",
|
||||
]
|
||||
}
|
||||
|
||||
issue_4943_2: {
|
||||
options = {
|
||||
reduce_vars: true,
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var a, b = 1;
|
||||
(function f() {
|
||||
a = "foo";
|
||||
b-- && f();
|
||||
console.log(a);
|
||||
a = "bar";
|
||||
})();
|
||||
}
|
||||
expect: {
|
||||
var a, b = 1;
|
||||
(function f() {
|
||||
a = "foo";
|
||||
b-- && f();
|
||||
console.log(a);
|
||||
a = "bar";
|
||||
})();
|
||||
}
|
||||
expect_stdout: [
|
||||
"foo",
|
||||
"bar",
|
||||
]
|
||||
}
|
||||
|
||||
issue_4949: {
|
||||
options = {
|
||||
reduce_vars: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
(function f(a) {
|
||||
a = 0;
|
||||
console.log(a++, arguments[0]);
|
||||
})(0);
|
||||
}
|
||||
expect: {
|
||||
(function(a) {
|
||||
a = 0;
|
||||
console.log(a++, arguments[0]);
|
||||
})(0);
|
||||
}
|
||||
expect_stdout: "0 1"
|
||||
}
|
||||
|
||||
issue_5048: {
|
||||
options = {
|
||||
reduce_funcs: true,
|
||||
reduce_vars: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
console.log(function() {
|
||||
var a = function() {
|
||||
return a + 42;
|
||||
};
|
||||
}());
|
||||
}
|
||||
expect: {
|
||||
console.log(function() {}());
|
||||
}
|
||||
expect_stdout: "undefined"
|
||||
}
|
||||
|
||||
issue_5050: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
reduce_vars: true,
|
||||
toplevel: true,
|
||||
}
|
||||
input: {
|
||||
function f() {
|
||||
console.log(a);
|
||||
}
|
||||
this;
|
||||
var a = 1;
|
||||
f(console.log(2), f(), a = 3);
|
||||
}
|
||||
expect: {
|
||||
function f() {
|
||||
console.log(a);
|
||||
}
|
||||
this;
|
||||
var a = 1;
|
||||
f(console.log(2), f(), a = 3);
|
||||
}
|
||||
expect_stdout: [
|
||||
"2",
|
||||
"1",
|
||||
"3",
|
||||
]
|
||||
}
|
||||
|
||||
issue_5055_1: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
reduce_vars: true,
|
||||
toplevel: true,
|
||||
}
|
||||
input: {
|
||||
var a = "PASS";
|
||||
function f() {
|
||||
console.log(a || "FAIL");
|
||||
}
|
||||
f(0 && (a = 0)(f(this)));
|
||||
}
|
||||
expect: {
|
||||
var a = "PASS";
|
||||
function f() {
|
||||
console.log(a || "FAIL");
|
||||
}
|
||||
f(0);
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
issue_5055_2: {
|
||||
options = {
|
||||
reduce_vars: true,
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var a = "PASS";
|
||||
function f() {
|
||||
console.log(a || "FAIL");
|
||||
}
|
||||
f(0 && (a = 0)(f(this)));
|
||||
}
|
||||
expect: {
|
||||
var a = "PASS";
|
||||
function f() {
|
||||
console.log(a || "FAIL");
|
||||
}
|
||||
f(0 && (a = 0)(f()));
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
@@ -1,11 +1,11 @@
|
||||
mangle_catch: {
|
||||
rename = true
|
||||
options = {
|
||||
ie8: false,
|
||||
ie: false,
|
||||
toplevel: false,
|
||||
}
|
||||
mangle = {
|
||||
ie8: false,
|
||||
ie: false,
|
||||
toplevel: false,
|
||||
}
|
||||
input: {
|
||||
@@ -24,11 +24,11 @@ mangle_catch: {
|
||||
mangle_catch_ie8: {
|
||||
rename = true
|
||||
options = {
|
||||
ie8: true,
|
||||
ie: true,
|
||||
toplevel: false,
|
||||
}
|
||||
mangle = {
|
||||
ie8: true,
|
||||
ie: true,
|
||||
toplevel: false,
|
||||
}
|
||||
input: {
|
||||
@@ -47,11 +47,11 @@ mangle_catch_ie8: {
|
||||
mangle_catch_var: {
|
||||
rename = true
|
||||
options = {
|
||||
ie8: false,
|
||||
ie: false,
|
||||
toplevel: false,
|
||||
}
|
||||
mangle = {
|
||||
ie8: false,
|
||||
ie: false,
|
||||
toplevel: false,
|
||||
}
|
||||
input: {
|
||||
@@ -70,11 +70,11 @@ mangle_catch_var: {
|
||||
mangle_catch_var_ie8: {
|
||||
rename = true
|
||||
options = {
|
||||
ie8: true,
|
||||
ie: true,
|
||||
toplevel: false,
|
||||
}
|
||||
mangle = {
|
||||
ie8: true,
|
||||
ie: true,
|
||||
toplevel: false,
|
||||
}
|
||||
input: {
|
||||
@@ -93,11 +93,11 @@ mangle_catch_var_ie8: {
|
||||
mangle_catch_toplevel: {
|
||||
rename = true
|
||||
options = {
|
||||
ie8: false,
|
||||
ie: false,
|
||||
toplevel: true,
|
||||
}
|
||||
mangle = {
|
||||
ie8: false,
|
||||
ie: false,
|
||||
toplevel: true,
|
||||
}
|
||||
input: {
|
||||
@@ -116,11 +116,11 @@ mangle_catch_toplevel: {
|
||||
mangle_catch_ie8_toplevel: {
|
||||
rename = true
|
||||
options = {
|
||||
ie8: true,
|
||||
ie: true,
|
||||
toplevel: true,
|
||||
}
|
||||
mangle = {
|
||||
ie8: true,
|
||||
ie: true,
|
||||
toplevel: true,
|
||||
}
|
||||
input: {
|
||||
@@ -139,11 +139,11 @@ mangle_catch_ie8_toplevel: {
|
||||
mangle_catch_var_toplevel: {
|
||||
rename = true
|
||||
options = {
|
||||
ie8: false,
|
||||
ie: false,
|
||||
toplevel: true,
|
||||
}
|
||||
mangle = {
|
||||
ie8: false,
|
||||
ie: false,
|
||||
toplevel: true,
|
||||
}
|
||||
input: {
|
||||
@@ -162,11 +162,11 @@ mangle_catch_var_toplevel: {
|
||||
mangle_catch_var_ie8_toplevel: {
|
||||
rename = true
|
||||
options = {
|
||||
ie8: true,
|
||||
ie: true,
|
||||
toplevel: true,
|
||||
}
|
||||
mangle = {
|
||||
ie8: true,
|
||||
ie: true,
|
||||
toplevel: true,
|
||||
}
|
||||
input: {
|
||||
@@ -185,11 +185,11 @@ mangle_catch_var_ie8_toplevel: {
|
||||
mangle_catch_redef_1: {
|
||||
rename = true
|
||||
options = {
|
||||
ie8: false,
|
||||
ie: false,
|
||||
toplevel: false,
|
||||
}
|
||||
mangle = {
|
||||
ie8: false,
|
||||
ie: false,
|
||||
toplevel: false,
|
||||
}
|
||||
input: {
|
||||
@@ -208,11 +208,11 @@ mangle_catch_redef_1: {
|
||||
mangle_catch_redef_1_ie8: {
|
||||
rename = true
|
||||
options = {
|
||||
ie8: true,
|
||||
ie: true,
|
||||
toplevel: false,
|
||||
}
|
||||
mangle = {
|
||||
ie8: true,
|
||||
ie: true,
|
||||
toplevel: false,
|
||||
}
|
||||
input: {
|
||||
@@ -231,11 +231,11 @@ mangle_catch_redef_1_ie8: {
|
||||
mangle_catch_redef_1_toplevel: {
|
||||
rename = true
|
||||
options = {
|
||||
ie8: false,
|
||||
ie: false,
|
||||
toplevel: true,
|
||||
}
|
||||
mangle = {
|
||||
ie8: false,
|
||||
ie: false,
|
||||
toplevel: true,
|
||||
}
|
||||
input: {
|
||||
@@ -254,11 +254,11 @@ mangle_catch_redef_1_toplevel: {
|
||||
mangle_catch_redef_1_ie8_toplevel: {
|
||||
rename = true
|
||||
options = {
|
||||
ie8: true,
|
||||
ie: true,
|
||||
toplevel: true,
|
||||
}
|
||||
mangle = {
|
||||
ie8: true,
|
||||
ie: true,
|
||||
toplevel: true,
|
||||
}
|
||||
input: {
|
||||
@@ -277,11 +277,11 @@ mangle_catch_redef_1_ie8_toplevel: {
|
||||
mangle_catch_redef_2: {
|
||||
rename = true
|
||||
options = {
|
||||
ie8: false,
|
||||
ie: false,
|
||||
toplevel: false,
|
||||
}
|
||||
mangle = {
|
||||
ie8: false,
|
||||
ie: false,
|
||||
toplevel: false,
|
||||
}
|
||||
input: {
|
||||
@@ -299,11 +299,11 @@ mangle_catch_redef_2: {
|
||||
mangle_catch_redef_2_ie8: {
|
||||
rename = true
|
||||
options = {
|
||||
ie8: true,
|
||||
ie: true,
|
||||
toplevel: false,
|
||||
}
|
||||
mangle = {
|
||||
ie8: true,
|
||||
ie: true,
|
||||
toplevel: false,
|
||||
}
|
||||
input: {
|
||||
@@ -321,11 +321,11 @@ mangle_catch_redef_2_ie8: {
|
||||
mangle_catch_redef_2_toplevel: {
|
||||
rename = true
|
||||
options = {
|
||||
ie8: false,
|
||||
ie: false,
|
||||
toplevel: true,
|
||||
}
|
||||
mangle = {
|
||||
ie8: false,
|
||||
ie: false,
|
||||
toplevel: true,
|
||||
}
|
||||
input: {
|
||||
@@ -343,11 +343,11 @@ mangle_catch_redef_2_toplevel: {
|
||||
mangle_catch_redef_2_ie8_toplevel: {
|
||||
rename = true
|
||||
options = {
|
||||
ie8: true,
|
||||
ie: true,
|
||||
toplevel: true,
|
||||
}
|
||||
mangle = {
|
||||
ie8: true,
|
||||
ie: true,
|
||||
toplevel: true,
|
||||
}
|
||||
input: {
|
||||
@@ -365,7 +365,7 @@ mangle_catch_redef_2_ie8_toplevel: {
|
||||
issue_2120_1: {
|
||||
rename = true
|
||||
mangle = {
|
||||
ie8: false,
|
||||
ie: false,
|
||||
}
|
||||
input: {
|
||||
"aaaaaaaa";
|
||||
@@ -401,7 +401,7 @@ issue_2120_1: {
|
||||
issue_2120_2: {
|
||||
rename = true
|
||||
mangle = {
|
||||
ie8: true,
|
||||
ie: true,
|
||||
}
|
||||
input: {
|
||||
"aaaaaaaa";
|
||||
@@ -436,7 +436,7 @@ issue_2120_2: {
|
||||
function_iife_catch: {
|
||||
rename = true
|
||||
mangle = {
|
||||
ie8: false,
|
||||
ie: false,
|
||||
}
|
||||
input: {
|
||||
function f(n) {
|
||||
@@ -458,7 +458,7 @@ function_iife_catch: {
|
||||
function_iife_catch_ie8: {
|
||||
rename = true
|
||||
mangle = {
|
||||
ie8: true,
|
||||
ie: true,
|
||||
}
|
||||
input: {
|
||||
function f(n) {
|
||||
@@ -480,7 +480,7 @@ function_iife_catch_ie8: {
|
||||
function_catch_catch: {
|
||||
rename = true
|
||||
mangle = {
|
||||
ie8: false,
|
||||
ie: false,
|
||||
}
|
||||
input: {
|
||||
var o = 0;
|
||||
@@ -509,7 +509,7 @@ function_catch_catch: {
|
||||
function_catch_catch_ie8: {
|
||||
rename = true
|
||||
mangle = {
|
||||
ie8: true,
|
||||
ie: true,
|
||||
}
|
||||
input: {
|
||||
var o = 0;
|
||||
@@ -538,12 +538,12 @@ function_catch_catch_ie8: {
|
||||
function_do_catch_ie8: {
|
||||
rename = true
|
||||
options = {
|
||||
ie8: true,
|
||||
ie: true,
|
||||
side_effects: true,
|
||||
unused: true,
|
||||
}
|
||||
mangle = {
|
||||
ie8: true,
|
||||
ie: true,
|
||||
toplevel: true,
|
||||
}
|
||||
input: {
|
||||
@@ -615,7 +615,7 @@ function_do_catch_ie8: {
|
||||
issue_3480: {
|
||||
rename = true,
|
||||
mangle = {
|
||||
ie8: false,
|
||||
ie: false,
|
||||
toplevel: false,
|
||||
}
|
||||
input: {
|
||||
@@ -647,7 +647,7 @@ issue_3480: {
|
||||
issue_3480_ie8: {
|
||||
rename = true,
|
||||
mangle = {
|
||||
ie8: true,
|
||||
ie: true,
|
||||
toplevel: false,
|
||||
}
|
||||
input: {
|
||||
@@ -679,7 +679,7 @@ issue_3480_ie8: {
|
||||
issue_3480_toplevel: {
|
||||
rename = true,
|
||||
mangle = {
|
||||
ie8: false,
|
||||
ie: false,
|
||||
toplevel: true,
|
||||
}
|
||||
input: {
|
||||
@@ -711,7 +711,7 @@ issue_3480_toplevel: {
|
||||
issue_3480_ie8_toplevel: {
|
||||
rename = true,
|
||||
mangle = {
|
||||
ie8: true,
|
||||
ie: true,
|
||||
toplevel: true,
|
||||
}
|
||||
input: {
|
||||
|
||||
@@ -244,7 +244,7 @@ retain_destructured_array: {
|
||||
console.log.apply(console, b);
|
||||
}
|
||||
expect: {
|
||||
var [ , ...b ] = [ "FAIL", "PASS", 42 ];
|
||||
var [ ...b ] = [ "PASS", 42 ];
|
||||
console.log.apply(console, b);
|
||||
}
|
||||
expect_stdout: "PASS 42"
|
||||
@@ -284,7 +284,7 @@ retain_destructured_object_2: {
|
||||
console.log(k, b[k]);
|
||||
}
|
||||
expect: {
|
||||
var { foo: [], ...b } = { foo: [ "FAIL" ], bar: "PASS", baz: 42 };
|
||||
var { foo: {}, ...b } = { foo: 0, bar: "PASS", baz: 42 };
|
||||
for (var k in b)
|
||||
console.log(k, b[k]);
|
||||
}
|
||||
@@ -407,6 +407,24 @@ drop_unused_call_args_2: {
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
maintain_position: {
|
||||
options = {
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
A = "FAIL";
|
||||
var [ , ...a ] = [ A, "PASS" ];
|
||||
console.log(a[0]);
|
||||
}
|
||||
expect: {
|
||||
A = "FAIL";
|
||||
var [ , ...a ] = [ A, "PASS" ];
|
||||
console.log(a[0]);
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
merge_funarg: {
|
||||
options = {
|
||||
merge_vars: true,
|
||||
@@ -491,7 +509,7 @@ drop_rest_array: {
|
||||
rests: true,
|
||||
}
|
||||
input: {
|
||||
var [ ...[ a ]] = [ "PASS" ];
|
||||
var [ ...[ a ] ] = [ "PASS" ];
|
||||
console.log(a);
|
||||
}
|
||||
expect: {
|
||||
@@ -542,6 +560,100 @@ drop_rest_lambda: {
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
keep_rest_array: {
|
||||
options = {
|
||||
rests: true,
|
||||
}
|
||||
input: {
|
||||
var [ ...[ ...a ] ] = "PASS";
|
||||
console.log(a.join(""));
|
||||
}
|
||||
expect: {
|
||||
var [ ...a ] = "PASS";
|
||||
console.log(a.join(""));
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
keep_rest_arrow: {
|
||||
options = {
|
||||
arrows: true,
|
||||
keep_fargs: false,
|
||||
reduce_vars: true,
|
||||
rests: true,
|
||||
}
|
||||
input: {
|
||||
console.log(((...[ ...a ]) => a.join(""))("PASS"));
|
||||
}
|
||||
expect: {
|
||||
console.log(((...a) => a.join(""))("PASS"));
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
keep_rest_lambda_1: {
|
||||
options = {
|
||||
keep_fargs: false,
|
||||
reduce_vars: true,
|
||||
rests: true,
|
||||
toplevel: true,
|
||||
}
|
||||
input: {
|
||||
function f(...[ ...a ]) {
|
||||
return a.join("");
|
||||
}
|
||||
console.log(f("PASS"), f([ 42 ]));
|
||||
}
|
||||
expect: {
|
||||
function f(...a) {
|
||||
return a.join("");
|
||||
}
|
||||
console.log(f("PASS"), f([ 42 ]));
|
||||
}
|
||||
expect_stdout: "PASS 42"
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
keep_rest_lambda_2: {
|
||||
options = {
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
function f(...[ ...a ]) {
|
||||
return a.join("");
|
||||
}
|
||||
console.log(f("PASS"), f([ 42 ]));
|
||||
}
|
||||
expect: {
|
||||
function f(...[ ...a ]) {
|
||||
return a.join("");
|
||||
}
|
||||
console.log(f("PASS"), f([ 42 ]));
|
||||
}
|
||||
expect_stdout: "PASS 42"
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
drop_new_function: {
|
||||
options = {
|
||||
side_effects: true,
|
||||
}
|
||||
input: {
|
||||
new function(...{
|
||||
[console.log("PASS")]: a,
|
||||
}) {}();
|
||||
}
|
||||
expect: {
|
||||
void ([ ... {
|
||||
[console.log("PASS")]: [].e,
|
||||
}] = []);
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
issue_4525_1: {
|
||||
options = {
|
||||
arguments: true,
|
||||
@@ -647,6 +759,78 @@ issue_4544_2: {
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
issue_4560_1: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
reduce_vars: true,
|
||||
toplevel: true,
|
||||
}
|
||||
input: {
|
||||
var a = 0;
|
||||
(function(...{
|
||||
[a++]: {},
|
||||
}) {})(2);
|
||||
console.log(a);
|
||||
}
|
||||
expect: {
|
||||
var a = 0;
|
||||
(function(...{
|
||||
[a++]: {},
|
||||
}) {})(2);
|
||||
console.log(a);
|
||||
}
|
||||
expect_stdout: "1"
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
issue_4560_2: {
|
||||
options = {
|
||||
reduce_vars: true,
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var a = 0;
|
||||
(function(...{
|
||||
[a++]: {},
|
||||
}) {})(2);
|
||||
console.log(a);
|
||||
}
|
||||
expect: {
|
||||
var a = 0;
|
||||
(function(...{
|
||||
[a++]: {},
|
||||
}) {})(2);
|
||||
console.log(a);
|
||||
}
|
||||
expect_stdout: "1"
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
issue_4560_3: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
reduce_vars: true,
|
||||
toplevel: true,
|
||||
}
|
||||
input: {
|
||||
var a = 0, b;
|
||||
[ ...{
|
||||
[a++]: b,
|
||||
} ] = [ "PASS" ];
|
||||
console.log(b);
|
||||
}
|
||||
expect: {
|
||||
var a = 0, b;
|
||||
[ ...{
|
||||
[a++]: b,
|
||||
} ] = [ "PASS" ];
|
||||
console.log(b);
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
issue_4562: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
@@ -667,7 +851,7 @@ issue_4562: {
|
||||
issue_4575: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
ie8: true,
|
||||
ie: true,
|
||||
reduce_vars: true,
|
||||
rests: true,
|
||||
unused: true,
|
||||
@@ -777,10 +961,151 @@ issue_4666: {
|
||||
expect: {
|
||||
var a = 0, b = 0;
|
||||
var o = (c => +a + c)([ b ]);
|
||||
for(var k in o)
|
||||
for (var k in o)
|
||||
b++;
|
||||
console.log(1, b);
|
||||
}
|
||||
expect_stdout: "1 2"
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
issue_5089_1: {
|
||||
options = {
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var {
|
||||
p: [] = 42,
|
||||
...o
|
||||
} = {
|
||||
p: [],
|
||||
};
|
||||
console.log(o.p);
|
||||
}
|
||||
expect: {
|
||||
var {
|
||||
p: {},
|
||||
...o
|
||||
} = {
|
||||
p: 0,
|
||||
};
|
||||
console.log(o.p);
|
||||
}
|
||||
expect_stdout: "undefined"
|
||||
node_version: ">=8"
|
||||
}
|
||||
|
||||
issue_5089_2: {
|
||||
options = {
|
||||
pure_getters: "strict",
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var {
|
||||
p: {} = null,
|
||||
...o
|
||||
} = {
|
||||
p: {},
|
||||
};
|
||||
console.log(o.p);
|
||||
}
|
||||
expect: {
|
||||
var {
|
||||
p: {},
|
||||
...o
|
||||
} = {
|
||||
p: 0,
|
||||
};
|
||||
console.log(o.p);
|
||||
}
|
||||
expect_stdout: "undefined"
|
||||
node_version: ">=8"
|
||||
}
|
||||
|
||||
issue_5100_1: {
|
||||
options = {
|
||||
passes: 2,
|
||||
pure_getters: "strict",
|
||||
side_effects: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var a;
|
||||
[ {
|
||||
p: {},
|
||||
...a
|
||||
} ] = [ {
|
||||
p: {
|
||||
q: a,
|
||||
} = 42,
|
||||
r: "PASS",
|
||||
} ];
|
||||
console.log(a.r);
|
||||
}
|
||||
expect: {
|
||||
var a;
|
||||
[ {
|
||||
p: {},
|
||||
...a
|
||||
} ] = [ {
|
||||
p: [ a = 42["q"] ],
|
||||
r: "PASS",
|
||||
} ];
|
||||
console.log(a.r);
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=8"
|
||||
}
|
||||
|
||||
issue_5100_2: {
|
||||
options = {
|
||||
passes: 2,
|
||||
pure_getters: "strict",
|
||||
side_effects: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var a;
|
||||
[ {
|
||||
p: {},
|
||||
...a
|
||||
} ] = [ {
|
||||
p: (console.log("PASS"), {
|
||||
q: a,
|
||||
} = 42),
|
||||
} ];
|
||||
}
|
||||
expect: {
|
||||
var a;
|
||||
[ {
|
||||
p: {},
|
||||
...a
|
||||
} ] = [ {
|
||||
p: [ console.log("PASS"), a = 42["q"] ],
|
||||
} ];
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=10"
|
||||
}
|
||||
|
||||
issue_5108: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
reduce_vars: true,
|
||||
rests: true,
|
||||
unsafe: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
console.log(function([ ...[ a ] ]) {
|
||||
return a;
|
||||
}([ "PASS", "FAIL" ]));
|
||||
}
|
||||
expect: {
|
||||
console.log(function([]) {
|
||||
return "PASS";
|
||||
}([ "PASS", "FAIL" ]));
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
@@ -172,3 +172,32 @@ issue_4054: {
|
||||
}
|
||||
expect_stdout: "{ p: [Setter] }"
|
||||
}
|
||||
|
||||
issue_4811_1: {
|
||||
input: {
|
||||
for (var PASS in this);
|
||||
console.log(PASS, this, {} < this);
|
||||
}
|
||||
expect: {
|
||||
for (var PASS in this);
|
||||
console.log(PASS, this, {} < this);
|
||||
}
|
||||
expect_stdout: "PASS [object global] true"
|
||||
}
|
||||
|
||||
issue_4811_2: {
|
||||
options = {
|
||||
side_effects: true,
|
||||
}
|
||||
input: {
|
||||
(async function() {});
|
||||
for (var PASS in this);
|
||||
console.log(PASS, this, {} < this);
|
||||
}
|
||||
expect: {
|
||||
for (var PASS in this);
|
||||
console.log(PASS, this, {} < this);
|
||||
}
|
||||
expect_stdout: "PASS [object global] true"
|
||||
node_version: ">=8"
|
||||
}
|
||||
|
||||
@@ -564,6 +564,34 @@ delete_seq_3: {
|
||||
}
|
||||
|
||||
delete_seq_4: {
|
||||
options = {
|
||||
booleans: true,
|
||||
evaluate: false,
|
||||
sequences: true,
|
||||
side_effects: true,
|
||||
}
|
||||
input: {
|
||||
function f() {}
|
||||
console.log(delete (f(), undefined));
|
||||
console.log(delete (f(), void 0));
|
||||
console.log(delete (f(), Infinity));
|
||||
console.log(delete (f(), 1 / 0));
|
||||
console.log(delete (f(), NaN));
|
||||
console.log(delete (f(), 0 / 0));
|
||||
}
|
||||
expect: {
|
||||
function f() {}
|
||||
console.log(delete void f()),
|
||||
console.log(delete void f()),
|
||||
console.log((f(), delete (1 / 0))),
|
||||
console.log((f(), delete (1 / 0))),
|
||||
console.log(delete (f(), NaN)),
|
||||
console.log((f(), delete(0 / 0)));
|
||||
}
|
||||
expect_stdout: true
|
||||
}
|
||||
|
||||
delete_seq_4_evaluate: {
|
||||
options = {
|
||||
booleans: true,
|
||||
evaluate: true,
|
||||
@@ -592,6 +620,35 @@ delete_seq_4: {
|
||||
}
|
||||
|
||||
delete_seq_5: {
|
||||
options = {
|
||||
booleans: true,
|
||||
evaluate: false,
|
||||
keep_infinity: true,
|
||||
sequences: true,
|
||||
side_effects: true,
|
||||
}
|
||||
input: {
|
||||
function f() {}
|
||||
console.log(delete (f(), undefined));
|
||||
console.log(delete (f(), void 0));
|
||||
console.log(delete (f(), Infinity));
|
||||
console.log(delete (f(), 1 / 0));
|
||||
console.log(delete (f(), NaN));
|
||||
console.log(delete (f(), 0 / 0));
|
||||
}
|
||||
expect: {
|
||||
function f() {}
|
||||
console.log(delete void f()),
|
||||
console.log(delete void f()),
|
||||
console.log(delete (f(), Infinity)),
|
||||
console.log((f(), delete (1 / 0))),
|
||||
console.log(delete (f(), NaN)),
|
||||
console.log((f(), delete (0 / 0)));
|
||||
}
|
||||
expect_stdout: true
|
||||
}
|
||||
|
||||
delete_seq_5_evaluate: {
|
||||
options = {
|
||||
booleans: true,
|
||||
evaluate: true,
|
||||
|
||||
@@ -198,6 +198,36 @@ global_fns: {
|
||||
]
|
||||
}
|
||||
|
||||
global_constructors: {
|
||||
options = {
|
||||
side_effects: true,
|
||||
unsafe: true,
|
||||
}
|
||||
input: {
|
||||
Map;
|
||||
new Map(console.log("foo"));
|
||||
Set;
|
||||
new Set(console.log("bar"));
|
||||
WeakMap;
|
||||
new WeakMap(console.log("baz"));
|
||||
WeakSet;
|
||||
new WeakSet(console.log("moo"));
|
||||
}
|
||||
expect: {
|
||||
console.log("foo");
|
||||
console.log("bar");
|
||||
console.log("baz");
|
||||
console.log("moo");
|
||||
}
|
||||
expect_stdout: [
|
||||
"foo",
|
||||
"bar",
|
||||
"baz",
|
||||
"moo",
|
||||
]
|
||||
node_version: ">=0.12"
|
||||
}
|
||||
|
||||
unsafe_builtin_1: {
|
||||
options = {
|
||||
side_effects: true,
|
||||
|
||||
@@ -85,7 +85,7 @@ collapse_vars_4: {
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
conditionals_farg: {
|
||||
conditionals_farg_1: {
|
||||
options = {
|
||||
conditionals: true,
|
||||
}
|
||||
@@ -107,6 +107,28 @@ conditionals_farg: {
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
conditionals_farg_2: {
|
||||
options = {
|
||||
conditionals: true,
|
||||
pure_getters: "strict",
|
||||
reduce_vars: true,
|
||||
}
|
||||
input: {
|
||||
var log = console.log;
|
||||
(function(a) {
|
||||
return a.length ? log(...a) : log("FAIL");
|
||||
})([ "PASS" ]);
|
||||
}
|
||||
expect: {
|
||||
var log = console.log;
|
||||
(function(a) {
|
||||
return a.length ? log(...a) : log("FAIL");
|
||||
})([ "PASS" ]);
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
dont_inline: {
|
||||
options = {
|
||||
inline: true,
|
||||
@@ -272,6 +294,31 @@ reduce_vars_2: {
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
reduce_vars_3: {
|
||||
options = {
|
||||
reduce_funcs: true,
|
||||
reduce_vars: true,
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
function f() {}
|
||||
function g() {
|
||||
return (a => a)(...[ f ]);
|
||||
}
|
||||
console.log(g() === g() ? "PASS" : "FAIL");
|
||||
}
|
||||
expect: {
|
||||
function f() {}
|
||||
function g() {
|
||||
return (a => a)(...[ f ]);
|
||||
}
|
||||
console.log(g() === g() ? "PASS" : "FAIL");
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
convert_setter: {
|
||||
options = {
|
||||
objects: true,
|
||||
@@ -667,6 +714,57 @@ unused_var_side_effects: {
|
||||
node_version: ">=8"
|
||||
}
|
||||
|
||||
unsafe_join_1: {
|
||||
options = {
|
||||
unsafe: true,
|
||||
}
|
||||
input: {
|
||||
console.log([ ..."foo" ].join());
|
||||
}
|
||||
expect: {
|
||||
console.log([ ..."foo" ].join());
|
||||
}
|
||||
expect_stdout: "f,o,o"
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
unsafe_join_2: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
unsafe: true,
|
||||
}
|
||||
input: {
|
||||
console.log([ "foo", ..."bar" ].join(""));
|
||||
}
|
||||
expect: {
|
||||
console.log([ "foo", ..."bar" ].join(""));
|
||||
}
|
||||
expect_stdout: "foobar"
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
unsafe_join_3: {
|
||||
options = {
|
||||
unsafe: true,
|
||||
}
|
||||
input: {
|
||||
try {
|
||||
[].join(...console);
|
||||
} catch (e) {
|
||||
console.log("PASS");
|
||||
}
|
||||
}
|
||||
expect: {
|
||||
try {
|
||||
[].join(...console);
|
||||
} catch (e) {
|
||||
console.log("PASS");
|
||||
}
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
issue_4329: {
|
||||
options = {
|
||||
objects: true,
|
||||
@@ -847,78 +945,6 @@ issue_4556: {
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
issue_4560_1: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
reduce_vars: true,
|
||||
toplevel: true,
|
||||
}
|
||||
input: {
|
||||
var a = 0;
|
||||
(function(...{
|
||||
[a++]: {},
|
||||
}) {})(2);
|
||||
console.log(a);
|
||||
}
|
||||
expect: {
|
||||
var a = 0;
|
||||
(function(...{
|
||||
[a++]: {},
|
||||
}) {})(2);
|
||||
console.log(a);
|
||||
}
|
||||
expect_stdout: "1"
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
issue_4560_2: {
|
||||
options = {
|
||||
reduce_vars: true,
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var a = 0;
|
||||
(function(...{
|
||||
[a++]: {},
|
||||
}) {})(2);
|
||||
console.log(a);
|
||||
}
|
||||
expect: {
|
||||
var a = 0;
|
||||
(function(...{
|
||||
[a++]: {},
|
||||
}) {})(2);
|
||||
console.log(a);
|
||||
}
|
||||
expect_stdout: "1"
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
issue_4560_3: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
reduce_vars: true,
|
||||
toplevel: true,
|
||||
}
|
||||
input: {
|
||||
var a = 0, b;
|
||||
[ ...{
|
||||
[a++]: b,
|
||||
} ] = [ "PASS" ];
|
||||
console.log(b);
|
||||
}
|
||||
expect: {
|
||||
var a = 0, b;
|
||||
[ ...{
|
||||
[a++]: b,
|
||||
} ] = [ "PASS" ];
|
||||
console.log(b);
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
issue_4614: {
|
||||
options = {
|
||||
pure_getters: "strict",
|
||||
@@ -947,3 +973,134 @@ issue_4614: {
|
||||
expect_stdout: true
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
issue_4849: {
|
||||
options = {
|
||||
reduce_vars: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
while (function() {
|
||||
while (!console);
|
||||
}(new function(a) {
|
||||
console.log(typeof { ...a });
|
||||
}(function() {})));
|
||||
}
|
||||
expect: {
|
||||
while (function() {
|
||||
while (!console);
|
||||
}(function(a) {
|
||||
console.log(typeof { ...function() {} });
|
||||
}()));
|
||||
}
|
||||
expect_stdout: "object"
|
||||
node_version: ">=8"
|
||||
}
|
||||
|
||||
issue_4882_1: {
|
||||
options = {
|
||||
objects: true,
|
||||
spreads: true,
|
||||
}
|
||||
input: {
|
||||
var o = {
|
||||
p: "PASS",
|
||||
... {
|
||||
__proto__: {
|
||||
p: "FAIL 1",
|
||||
q: "FAIL 2",
|
||||
},
|
||||
},
|
||||
};
|
||||
console.log(o.p);
|
||||
console.log(o.q);
|
||||
}
|
||||
expect: {
|
||||
var o = {
|
||||
p: "PASS",
|
||||
};
|
||||
console.log(o.p);
|
||||
console.log(o.q);
|
||||
}
|
||||
expect_stdout: [
|
||||
"PASS",
|
||||
"undefined",
|
||||
]
|
||||
node_version: ">=8"
|
||||
}
|
||||
|
||||
issue_4882_2: {
|
||||
options = {
|
||||
objects: true,
|
||||
spreads: true,
|
||||
}
|
||||
input: {
|
||||
console.log(null == Object.getPrototypeOf({
|
||||
... {
|
||||
__proto__: (console.log(42), null),
|
||||
},
|
||||
}) ? "FAIL" : "PASS");
|
||||
}
|
||||
expect: {
|
||||
console.log(null == Object.getPrototypeOf({
|
||||
... {
|
||||
__proto__: (console.log(42), null),
|
||||
},
|
||||
}) ? "FAIL" : "PASS");
|
||||
}
|
||||
expect_stdout: [
|
||||
"42",
|
||||
"PASS",
|
||||
]
|
||||
node_version: ">=8"
|
||||
}
|
||||
|
||||
issue_4882_3: {
|
||||
options = {
|
||||
objects: true,
|
||||
spreads: true,
|
||||
}
|
||||
input: {
|
||||
var o = {
|
||||
__proto__: { p: 42 },
|
||||
... {
|
||||
set __proto__(v) {},
|
||||
},
|
||||
};
|
||||
console.log(o.__proto__ === Object.getPrototypeOf(o) ? "FAIL" : "PASS");
|
||||
console.log(o.p);
|
||||
}
|
||||
expect: {
|
||||
var o = {
|
||||
__proto__: { p: 42 },
|
||||
["__proto__"]: void 0,
|
||||
};
|
||||
console.log(o.__proto__ === Object.getPrototypeOf(o) ? "FAIL" : "PASS");
|
||||
console.log(o.p);
|
||||
}
|
||||
expect_stdout: [
|
||||
"PASS",
|
||||
"42",
|
||||
]
|
||||
node_version: ">=8"
|
||||
}
|
||||
|
||||
issue_5006: {
|
||||
options = {
|
||||
arguments: true,
|
||||
}
|
||||
input: {
|
||||
console.log(function(b, c) {
|
||||
c = "FAIL 2";
|
||||
return arguments[1];
|
||||
}(...[], "FAIL 1") || "PASS");
|
||||
}
|
||||
expect: {
|
||||
console.log(function(b, c) {
|
||||
c = "FAIL 2";
|
||||
return arguments[1];
|
||||
}(...[], "FAIL 1") || "PASS");
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
@@ -113,7 +113,7 @@ constant_switch_5: {
|
||||
// the break inside the if ruins our job
|
||||
// we can still get rid of irrelevant cases.
|
||||
switch (1) {
|
||||
case 1:
|
||||
default:
|
||||
x();
|
||||
if (foo) break;
|
||||
y();
|
||||
@@ -300,6 +300,37 @@ drop_default_2: {
|
||||
}
|
||||
}
|
||||
|
||||
drop_default_3: {
|
||||
options = {
|
||||
dead_code: true,
|
||||
evaluate: true,
|
||||
switches: true,
|
||||
}
|
||||
input: {
|
||||
function f() {
|
||||
console.log("PASS");
|
||||
return 42;
|
||||
}
|
||||
switch (42) {
|
||||
case f():
|
||||
break;
|
||||
case void console.log("FAIL"):
|
||||
default:
|
||||
}
|
||||
}
|
||||
expect: {
|
||||
function f() {
|
||||
console.log("PASS");
|
||||
return 42;
|
||||
}
|
||||
switch (42) {
|
||||
case f():
|
||||
case void console.log("FAIL"):
|
||||
}
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
keep_default: {
|
||||
options = {
|
||||
dead_code: true,
|
||||
@@ -423,7 +454,6 @@ drop_case_3: {
|
||||
switch ({}.p) {
|
||||
default:
|
||||
case void 0:
|
||||
break;
|
||||
case c = "FAIL":
|
||||
}
|
||||
console.log(c);
|
||||
@@ -454,7 +484,168 @@ drop_case_4: {
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
keep_case: {
|
||||
drop_case_5: {
|
||||
options = {
|
||||
dead_code: true,
|
||||
evaluate: true,
|
||||
switches: true,
|
||||
}
|
||||
input: {
|
||||
switch (42) {
|
||||
case void console.log("PASS 1"):
|
||||
console.log("FAIL 1");
|
||||
case 42:
|
||||
case console.log("FAIL 2"):
|
||||
console.log("PASS 2");
|
||||
}
|
||||
}
|
||||
expect: {
|
||||
switch (42) {
|
||||
default:
|
||||
void console.log("PASS 1");
|
||||
console.log("PASS 2");
|
||||
}
|
||||
}
|
||||
expect_stdout: [
|
||||
"PASS 1",
|
||||
"PASS 2",
|
||||
]
|
||||
}
|
||||
|
||||
drop_case_6: {
|
||||
options = {
|
||||
dead_code: true,
|
||||
evaluate: true,
|
||||
switches: true,
|
||||
}
|
||||
input: {
|
||||
switch (console.log("PASS 1"), 2) {
|
||||
case 0:
|
||||
console.log("FAIL 1");
|
||||
case (console.log("PASS 2"), 1):
|
||||
console.log("FAIL 2");
|
||||
}
|
||||
}
|
||||
expect: {
|
||||
switch (console.log("PASS 1"), 2) {
|
||||
case (console.log("PASS 2"), 1):
|
||||
}
|
||||
}
|
||||
expect_stdout: [
|
||||
"PASS 1",
|
||||
"PASS 2",
|
||||
]
|
||||
}
|
||||
|
||||
drop_case_7: {
|
||||
options = {
|
||||
dead_code: true,
|
||||
evaluate: true,
|
||||
switches: true,
|
||||
}
|
||||
input: {
|
||||
switch (2) {
|
||||
case 0:
|
||||
console.log("FAIL 1");
|
||||
case (console.log("PASS 1"), 1):
|
||||
console.log("FAIL 2");
|
||||
case 2:
|
||||
console.log("PASS 2");
|
||||
}
|
||||
}
|
||||
expect: {
|
||||
switch (2) {
|
||||
default:
|
||||
console.log("PASS 1"), 1;
|
||||
console.log("PASS 2");
|
||||
}
|
||||
}
|
||||
expect_stdout: [
|
||||
"PASS 1",
|
||||
"PASS 2",
|
||||
]
|
||||
}
|
||||
|
||||
drop_case_8: {
|
||||
options = {
|
||||
dead_code: true,
|
||||
switches: true,
|
||||
}
|
||||
input: {
|
||||
function log(msg) {
|
||||
console.log(msg);
|
||||
return msg;
|
||||
}
|
||||
switch (log("foo")) {
|
||||
case "bar":
|
||||
log("moo");
|
||||
break;
|
||||
case log("baz"):
|
||||
log("moo");
|
||||
break;
|
||||
default:
|
||||
log("moo");
|
||||
}
|
||||
}
|
||||
expect: {
|
||||
function log(msg) {
|
||||
console.log(msg);
|
||||
return msg;
|
||||
}
|
||||
switch (log("foo")) {
|
||||
case "bar":
|
||||
case log("baz"):
|
||||
default:
|
||||
log("moo");
|
||||
}
|
||||
}
|
||||
expect_stdout: [
|
||||
"foo",
|
||||
"baz",
|
||||
"moo",
|
||||
]
|
||||
}
|
||||
|
||||
drop_case_9: {
|
||||
options = {
|
||||
dead_code: true,
|
||||
switches: true,
|
||||
}
|
||||
input: {
|
||||
function log(msg) {
|
||||
console.log(msg);
|
||||
return msg;
|
||||
}
|
||||
switch (log("foo")) {
|
||||
case log("bar"):
|
||||
log("moo");
|
||||
break;
|
||||
case "baz":
|
||||
log("moo");
|
||||
break;
|
||||
default:
|
||||
log("moo");
|
||||
}
|
||||
}
|
||||
expect: {
|
||||
function log(msg) {
|
||||
console.log(msg);
|
||||
return msg;
|
||||
}
|
||||
switch (log("foo")) {
|
||||
default:
|
||||
log("bar");
|
||||
log("moo");
|
||||
}
|
||||
}
|
||||
expect_stdout: [
|
||||
"foo",
|
||||
"bar",
|
||||
"moo",
|
||||
]
|
||||
}
|
||||
|
||||
keep_case_1: {
|
||||
options = {
|
||||
dead_code: true,
|
||||
switches: true,
|
||||
@@ -474,6 +665,76 @@ keep_case: {
|
||||
}
|
||||
}
|
||||
|
||||
keep_case_2: {
|
||||
options = {
|
||||
dead_code: true,
|
||||
evaluate: true,
|
||||
switches: true,
|
||||
}
|
||||
input: {
|
||||
switch ("foo") {
|
||||
case console.log("bar"):
|
||||
case console.log("baz"), "moo":
|
||||
}
|
||||
}
|
||||
expect: {
|
||||
switch ("foo") {
|
||||
case console.log("bar"):
|
||||
case console.log("baz"), "moo":
|
||||
}
|
||||
}
|
||||
expect_stdout: [
|
||||
"bar",
|
||||
"baz",
|
||||
]
|
||||
}
|
||||
|
||||
keep_case_3: {
|
||||
options = {
|
||||
dead_code: true,
|
||||
evaluate: true,
|
||||
switches: true,
|
||||
}
|
||||
input: {
|
||||
var a;
|
||||
switch (void console.log("PASS")) {
|
||||
case a:
|
||||
case console.log("FAIL"), 42:
|
||||
}
|
||||
}
|
||||
expect: {
|
||||
var a;
|
||||
switch (void console.log("PASS")) {
|
||||
case a:
|
||||
case console.log("FAIL"), 42:
|
||||
}
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
keep_case_4: {
|
||||
options = {
|
||||
dead_code: true,
|
||||
evaluate: true,
|
||||
switches: true,
|
||||
}
|
||||
input: {
|
||||
var a;
|
||||
switch (void console.log("PASS")) {
|
||||
case a:
|
||||
case void console.log("FAIL"):
|
||||
}
|
||||
}
|
||||
expect: {
|
||||
var a;
|
||||
switch (void console.log("PASS")) {
|
||||
case a:
|
||||
case void console.log("FAIL"):
|
||||
}
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
issue_376: {
|
||||
options = {
|
||||
dead_code: true,
|
||||
@@ -661,7 +922,7 @@ issue_1680_1: {
|
||||
case f(0):
|
||||
case f(1):
|
||||
f(2);
|
||||
case 2:
|
||||
default:
|
||||
f(5);
|
||||
}
|
||||
}
|
||||
@@ -924,7 +1185,6 @@ issue_2535: {
|
||||
}
|
||||
expect: {
|
||||
w(), 42;
|
||||
42;
|
||||
y();
|
||||
z();
|
||||
}
|
||||
@@ -950,7 +1210,6 @@ issue_1750: {
|
||||
expect: {
|
||||
var a = 0, b = 1;
|
||||
true;
|
||||
a, true;
|
||||
b = 2;
|
||||
console.log(a, b);
|
||||
}
|
||||
@@ -1088,7 +1347,8 @@ drop_switch_6: {
|
||||
}
|
||||
}
|
||||
expect: {
|
||||
A === B;
|
||||
A;
|
||||
B;
|
||||
x();
|
||||
C !== D;
|
||||
y();
|
||||
@@ -1181,3 +1441,170 @@ issue_4059: {
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
issue_5008_1: {
|
||||
options = {
|
||||
dead_code: true,
|
||||
evaluate: true,
|
||||
reduce_vars: true,
|
||||
switches: true,
|
||||
unsafe: true,
|
||||
}
|
||||
input: {
|
||||
console.log(function f() {
|
||||
switch (f) {
|
||||
case f:
|
||||
return "PASS";
|
||||
default:
|
||||
return "FAIL";
|
||||
}
|
||||
}());
|
||||
}
|
||||
expect: {
|
||||
console.log(function f() {
|
||||
switch (f) {
|
||||
default:
|
||||
return "PASS";
|
||||
}
|
||||
}());
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
issue_5008_2: {
|
||||
options = {
|
||||
dead_code: true,
|
||||
evaluate: true,
|
||||
reduce_vars: true,
|
||||
switches: true,
|
||||
unsafe: true,
|
||||
}
|
||||
input: {
|
||||
console.log(function(a) {
|
||||
switch (a) {
|
||||
case a:
|
||||
return "PASS";
|
||||
default:
|
||||
return "FAIL";
|
||||
}
|
||||
}([]));
|
||||
}
|
||||
expect: {
|
||||
console.log(function(a) {
|
||||
switch (a) {
|
||||
default:
|
||||
return "PASS";
|
||||
}
|
||||
}([]));
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
issue_5008_3: {
|
||||
options = {
|
||||
dead_code: true,
|
||||
evaluate: true,
|
||||
reduce_vars: true,
|
||||
switches: true,
|
||||
unsafe: true,
|
||||
}
|
||||
input: {
|
||||
console.log(function(a) {
|
||||
switch (a) {
|
||||
case a:
|
||||
return "PASS";
|
||||
default:
|
||||
return "FAIL";
|
||||
}
|
||||
}({}));
|
||||
}
|
||||
expect: {
|
||||
console.log(function(a) {
|
||||
switch (a) {
|
||||
default:
|
||||
return "PASS";
|
||||
}
|
||||
}({}));
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
issue_5008_4: {
|
||||
options = {
|
||||
dead_code: true,
|
||||
evaluate: true,
|
||||
reduce_vars: true,
|
||||
switches: true,
|
||||
}
|
||||
input: {
|
||||
console.log(function(a) {
|
||||
switch (a) {
|
||||
case a:
|
||||
return "PASS";
|
||||
default:
|
||||
return "FAIL";
|
||||
}
|
||||
}(/foo/));
|
||||
}
|
||||
expect: {
|
||||
console.log(function(a) {
|
||||
switch (a) {
|
||||
default:
|
||||
return "PASS";
|
||||
}
|
||||
}(/foo/));
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
issue_5010: {
|
||||
options = {
|
||||
dead_code: true,
|
||||
evaluate: true,
|
||||
switches: true,
|
||||
}
|
||||
input: {
|
||||
var a;
|
||||
switch (42) {
|
||||
case console.log("PASS"):
|
||||
case a:
|
||||
console.log("FAIL");
|
||||
case 42:
|
||||
}
|
||||
}
|
||||
expect: {
|
||||
var a;
|
||||
switch (42) {
|
||||
case console.log("PASS"):
|
||||
case a:
|
||||
console.log("FAIL");
|
||||
}
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
issue_5012: {
|
||||
options = {
|
||||
dead_code: true,
|
||||
evaluate: true,
|
||||
switches: true,
|
||||
}
|
||||
input: {
|
||||
switch (void 0) {
|
||||
case console.log("PASS"):
|
||||
break;
|
||||
case void 0:
|
||||
case 42:
|
||||
console.log("FAIL");
|
||||
}
|
||||
}
|
||||
expect: {
|
||||
switch (void 0) {
|
||||
case console.log("PASS"):
|
||||
break;
|
||||
default:
|
||||
console.log("FAIL");
|
||||
}
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
@@ -62,6 +62,23 @@ tag_parentheses_arrow: {
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
tag_parentheses_binary: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var f = function() {
|
||||
console.log("PASS");
|
||||
} || console
|
||||
f``;
|
||||
}
|
||||
expect_exact: '(function(){console.log("PASS")}||console)``;'
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
tag_parentheses_new: {
|
||||
input: {
|
||||
(new function() {
|
||||
@@ -73,6 +90,35 @@ tag_parentheses_new: {
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
tag_parentheses_sequence: {
|
||||
input: {
|
||||
var o = {
|
||||
f() {
|
||||
console.log(this === o ? "FAIL" : "PASS");
|
||||
},
|
||||
};
|
||||
(42, o.f)``;
|
||||
}
|
||||
expect_exact: 'var o={f(){console.log(this===o?"FAIL":"PASS")}};(42,o.f)``;'
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
tag_parentheses_unary: {
|
||||
input: {
|
||||
var a;
|
||||
try {
|
||||
(~a)``;
|
||||
(a++)``;
|
||||
} catch (e) {
|
||||
console.log("PASS");
|
||||
}
|
||||
}
|
||||
expect_exact: 'var a;try{(~a)``;(a++)``}catch(e){console.log("PASS")}'
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
malformed_escape: {
|
||||
input: {
|
||||
(function(s) {
|
||||
@@ -211,7 +257,7 @@ unsafe_evaluate: {
|
||||
node_version: ">=8"
|
||||
}
|
||||
|
||||
side_effects: {
|
||||
side_effects_1: {
|
||||
options = {
|
||||
side_effects: true,
|
||||
}
|
||||
@@ -228,6 +274,30 @@ side_effects: {
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
side_effects_2: {
|
||||
options = {
|
||||
side_effects: true,
|
||||
}
|
||||
input: {
|
||||
var o = {
|
||||
f() {
|
||||
console.log(this === o ? "FAIL" : "PASS");
|
||||
},
|
||||
};
|
||||
(42, o.f)``;
|
||||
}
|
||||
expect: {
|
||||
var o = {
|
||||
f() {
|
||||
console.log(this === o ? "FAIL" : "PASS");
|
||||
},
|
||||
};
|
||||
(0, o.f)``;
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
unsafe_side_effects: {
|
||||
options = {
|
||||
side_effects: true,
|
||||
@@ -245,6 +315,21 @@ unsafe_side_effects: {
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
pure_funcs: {
|
||||
options = {
|
||||
pure_funcs: "Math.random",
|
||||
side_effects: true,
|
||||
}
|
||||
input: {
|
||||
Math.random`${console.log("PASS")}`;
|
||||
}
|
||||
expect: {
|
||||
console.log("PASS");
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
issue_4604: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
@@ -305,7 +390,7 @@ issue_4676: {
|
||||
reduce_vars: true,
|
||||
templates: true,
|
||||
toplevel: true,
|
||||
unsafe:true,
|
||||
unsafe: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
@@ -328,3 +413,24 @@ issue_4676: {
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
issue_4931: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
templates: true,
|
||||
unsafe: true,
|
||||
}
|
||||
input: {
|
||||
console.log(String.raw`${typeof A} ${"\r"}`);
|
||||
console.log(String.raw`${"\\"} ${"`"}`);
|
||||
}
|
||||
expect: {
|
||||
console.log(String.raw`${typeof A} ${"\r"}`);
|
||||
console.log("\\ `");
|
||||
}
|
||||
expect_stdout: [
|
||||
"undefined \r",
|
||||
"\\ `",
|
||||
]
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
@@ -95,7 +95,7 @@ while_if_break: {
|
||||
}
|
||||
}
|
||||
expect: {
|
||||
for(; a && (b && c && d, !e););
|
||||
for (; a && (b && c && d, !e););
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
@@ -50,7 +50,7 @@ unicode_parse_variables: {
|
||||
}
|
||||
}
|
||||
|
||||
unicode_escaped_identifier: {
|
||||
unicode_escaped_identifier_1: {
|
||||
input: {
|
||||
var \u0061 = "\ud800\udc00";
|
||||
console.log(a);
|
||||
@@ -59,6 +59,17 @@ unicode_escaped_identifier: {
|
||||
expect_stdout: "\ud800\udc00"
|
||||
}
|
||||
|
||||
unicode_escaped_identifier_2: {
|
||||
input: {
|
||||
var \u{61} = "foo";
|
||||
var \u{10000} = "bar";
|
||||
console.log(a, \u{10000});
|
||||
}
|
||||
expect_exact: 'var a="foo";var \u{10000}="bar";console.log(a,\u{10000});'
|
||||
expect_stdout: "foo bar"
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
unicode_identifier_ascii_only: {
|
||||
beautify = {
|
||||
ascii_only: true,
|
||||
|
||||
@@ -523,3 +523,93 @@ default_init: {
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
issue_4933_1: {
|
||||
options = {
|
||||
reduce_vars: true,
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
varify: true,
|
||||
}
|
||||
input: {
|
||||
console.log(f());
|
||||
function f() {
|
||||
var a;
|
||||
for (console in a = [ f ]) {
|
||||
const b = a;
|
||||
}
|
||||
}
|
||||
}
|
||||
expect: {
|
||||
console.log(function f() {
|
||||
var a;
|
||||
for (console in a = [ f ]) {
|
||||
const b = a;
|
||||
}
|
||||
}());
|
||||
}
|
||||
expect_stdout: "undefined"
|
||||
}
|
||||
|
||||
issue_4933_2: {
|
||||
options = {
|
||||
passes: 2,
|
||||
reduce_vars: true,
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
varify: true,
|
||||
}
|
||||
input: {
|
||||
console.log(f());
|
||||
function f() {
|
||||
var a;
|
||||
for (console in a = [ f ]) {
|
||||
const b = a;
|
||||
}
|
||||
}
|
||||
}
|
||||
expect: {
|
||||
console.log(function f() {
|
||||
for (console in [ f ]);
|
||||
}());
|
||||
}
|
||||
expect_stdout: "undefined"
|
||||
}
|
||||
|
||||
issue_4954: {
|
||||
options = {
|
||||
functions: true,
|
||||
reduce_vars: true,
|
||||
unused: true,
|
||||
varify: true,
|
||||
}
|
||||
input: {
|
||||
"use strict";
|
||||
(function() {
|
||||
{
|
||||
let a = console;
|
||||
console.log(typeof a);
|
||||
}
|
||||
{
|
||||
let a = function() {};
|
||||
a && console.log(typeof a);
|
||||
}
|
||||
})();
|
||||
}
|
||||
expect: {
|
||||
"use strict";
|
||||
(function() {
|
||||
var a = console;
|
||||
console.log(typeof a);
|
||||
{
|
||||
let a = function() {};
|
||||
a && console.log(typeof a);
|
||||
}
|
||||
})();
|
||||
}
|
||||
expect_stdout: [
|
||||
"object",
|
||||
"function",
|
||||
]
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
@@ -61,7 +61,7 @@ lambda_name_mangle: {
|
||||
|
||||
lambda_name_mangle_ie8: {
|
||||
mangle = {
|
||||
ie8: true,
|
||||
ie: true,
|
||||
toplevel: true,
|
||||
}
|
||||
input: {
|
||||
@@ -95,7 +95,7 @@ function_name_mangle_ie8: {
|
||||
unused: true,
|
||||
}
|
||||
mangle = {
|
||||
ie8: true,
|
||||
ie: true,
|
||||
toplevel: true,
|
||||
}
|
||||
input: {
|
||||
|
||||
@@ -150,6 +150,27 @@ for_await_of: {
|
||||
node_version: ">=10"
|
||||
}
|
||||
|
||||
comment_newline: {
|
||||
beautify = {
|
||||
comments: "all",
|
||||
}
|
||||
input: {
|
||||
console.log(function*() {
|
||||
yield (
|
||||
/* */
|
||||
"PASS"
|
||||
);
|
||||
}().next().value);
|
||||
}
|
||||
expect_exact: [
|
||||
"console.log(function*(){",
|
||||
"/* */",
|
||||
'yield"PASS"}().next().value);',
|
||||
]
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
collapse_vars_1: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
@@ -247,6 +268,30 @@ collapse_vars_4: {
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
collapse_vars_5: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
}
|
||||
input: {
|
||||
var a = function* f(b, c) {
|
||||
b = yield c = b;
|
||||
console.log(c);
|
||||
}("PASS");
|
||||
a.next();
|
||||
a.next("FAIL");
|
||||
}
|
||||
expect: {
|
||||
var a = function* f(b, c) {
|
||||
b = yield c = b;
|
||||
console.log(c);
|
||||
}("PASS");
|
||||
a.next();
|
||||
a.next("FAIL");
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
collapse_property_lambda: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
@@ -388,14 +433,14 @@ functions: {
|
||||
function* b() {
|
||||
return !!b;
|
||||
}
|
||||
var c = function*(c) {
|
||||
function* c(c) {
|
||||
return c;
|
||||
};
|
||||
}
|
||||
if (yield* c(yield* b(yield* a()))) {
|
||||
function* d() {}
|
||||
function* e() {
|
||||
return typeof e;
|
||||
}
|
||||
var d = function*() {};
|
||||
var e = function* y() {
|
||||
return typeof y;
|
||||
};
|
||||
var f = function*(f) {
|
||||
return f;
|
||||
};
|
||||
@@ -446,9 +491,9 @@ functions_use_strict: {
|
||||
function* b() {
|
||||
return !!b;
|
||||
}
|
||||
var c = function*(c) {
|
||||
function* c(c) {
|
||||
return c;
|
||||
};
|
||||
}
|
||||
if (yield* c(yield* b(yield* a()))) {
|
||||
var d = function*() {};
|
||||
var e = function* y() {
|
||||
@@ -465,6 +510,54 @@ functions_use_strict: {
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
functions_anonymous: {
|
||||
options = {
|
||||
functions: true,
|
||||
reduce_vars: true,
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var yield = function*() {
|
||||
return "PASS";
|
||||
};
|
||||
console.log(yield().next(yield).value);
|
||||
}
|
||||
expect: {
|
||||
function* yield() {
|
||||
return "PASS";
|
||||
}
|
||||
console.log(yield().next(yield).value);
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
functions_inner_var: {
|
||||
options = {
|
||||
functions: true,
|
||||
reduce_vars: true,
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var yield = function* a() {
|
||||
var a;
|
||||
console.log(a, a);
|
||||
};
|
||||
yield().next(yield);
|
||||
}
|
||||
expect: {
|
||||
function* yield() {
|
||||
var a;
|
||||
console.log(a, a);
|
||||
}
|
||||
yield().next(yield);
|
||||
}
|
||||
expect_stdout: "undefined undefined"
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
negate_iife: {
|
||||
options = {
|
||||
negate_iife: true,
|
||||
@@ -950,3 +1043,235 @@ issue_4641_2: {
|
||||
]
|
||||
node_version: ">=10"
|
||||
}
|
||||
|
||||
issue_4769_1: {
|
||||
options = {
|
||||
side_effects: true,
|
||||
}
|
||||
input: {
|
||||
console.log(function*() {
|
||||
(function({} = yield => {}) {})();
|
||||
}().next().done);
|
||||
}
|
||||
expect: {
|
||||
console.log(function*() {
|
||||
(function({} = yield => {}) {})();
|
||||
}().next().done);
|
||||
}
|
||||
expect_stdout: "true"
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
issue_4769_2: {
|
||||
options = {
|
||||
inline: true,
|
||||
}
|
||||
input: {
|
||||
console.log(function*() {
|
||||
return function({} = yield => {}) {
|
||||
return "PASS";
|
||||
}();
|
||||
}().next().value);
|
||||
}
|
||||
expect: {
|
||||
console.log(function*() {
|
||||
return function({} = yield => {}) {
|
||||
return "PASS";
|
||||
}();
|
||||
}().next().value);
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
issue_5019_1: {
|
||||
options = {
|
||||
dead_code: true,
|
||||
}
|
||||
input: {
|
||||
(function(a) {
|
||||
return a = function*() {
|
||||
console.log(typeof a);
|
||||
}();
|
||||
})().next();
|
||||
}
|
||||
expect: {
|
||||
(function(a) {
|
||||
return a = function*() {
|
||||
console.log(typeof a);
|
||||
}();
|
||||
})().next();
|
||||
}
|
||||
expect_stdout: "object"
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
issue_5019_2: {
|
||||
options = {
|
||||
inline: true,
|
||||
toplevel: true,
|
||||
}
|
||||
input: {
|
||||
var a = [];
|
||||
for (var b in "foo")
|
||||
a.push(function(c) {
|
||||
return function*() {
|
||||
console.log(c);
|
||||
}();
|
||||
}(b));
|
||||
a.map(function(d) {
|
||||
return d.next();
|
||||
});
|
||||
}
|
||||
expect: {
|
||||
var a = [];
|
||||
for (var b in "foo")
|
||||
a.push(function(c) {
|
||||
return function*() {
|
||||
console.log(c);
|
||||
}();
|
||||
}(b));
|
||||
a.map(function(d) {
|
||||
return d.next();
|
||||
});
|
||||
}
|
||||
expect_stdout: [
|
||||
"0",
|
||||
"1",
|
||||
"2",
|
||||
]
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
issue_5032_normal: {
|
||||
options = {
|
||||
merge_vars: true,
|
||||
webkit: false,
|
||||
}
|
||||
input: {
|
||||
function log(value) {
|
||||
console.log(value);
|
||||
return value;
|
||||
}
|
||||
function *f(a) {
|
||||
var b = log(a), c = b;
|
||||
log(b);
|
||||
log(c);
|
||||
}
|
||||
f("PASS").next();
|
||||
}
|
||||
expect: {
|
||||
function log(value) {
|
||||
console.log(value);
|
||||
return value;
|
||||
}
|
||||
function *f(c) {
|
||||
var b = log(c), c = b;
|
||||
log(b);
|
||||
log(c);
|
||||
}
|
||||
f("PASS").next();
|
||||
}
|
||||
expect_stdout: [
|
||||
"PASS",
|
||||
"PASS",
|
||||
"PASS",
|
||||
]
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
issue_5032_webkit: {
|
||||
options = {
|
||||
merge_vars: true,
|
||||
webkit: true,
|
||||
}
|
||||
input: {
|
||||
function log(value) {
|
||||
console.log(value);
|
||||
return value;
|
||||
}
|
||||
function *f(a) {
|
||||
var b = log(a), c = b;
|
||||
log(b);
|
||||
log(c);
|
||||
}
|
||||
f("PASS").next();
|
||||
}
|
||||
expect: {
|
||||
function log(value) {
|
||||
console.log(value);
|
||||
return value;
|
||||
}
|
||||
function *f(a) {
|
||||
var b = log(a), c = b;
|
||||
log(b);
|
||||
log(c);
|
||||
}
|
||||
f("PASS").next();
|
||||
}
|
||||
expect_stdout: [
|
||||
"PASS",
|
||||
"PASS",
|
||||
"PASS",
|
||||
]
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
issue_5034: {
|
||||
options = {
|
||||
functions: true,
|
||||
reduce_vars: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
console.log(function() {
|
||||
var yield = function f() {
|
||||
return function*() {
|
||||
return f;
|
||||
};
|
||||
};
|
||||
return yield()().next().value === yield;
|
||||
}() ? "PASS" : "FAIL");
|
||||
}
|
||||
expect: {
|
||||
console.log(function() {
|
||||
var yield = function f() {
|
||||
return function*() {
|
||||
return f;
|
||||
};
|
||||
};
|
||||
return yield()().next().value === yield;
|
||||
}() ? "PASS" : "FAIL");
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
issue_5076: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
hoist_vars: true,
|
||||
passes: 2,
|
||||
pure_getters: "strict",
|
||||
side_effects: true,
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
yields: true,
|
||||
}
|
||||
input: {
|
||||
var a;
|
||||
console.log("PASS");
|
||||
var b = function*({
|
||||
p: {},
|
||||
}) {}({
|
||||
p: { a } = 42,
|
||||
});
|
||||
}
|
||||
expect: {
|
||||
var a;
|
||||
console.log("PASS");
|
||||
a = 42["a"];
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
@@ -1 +1 @@
|
||||
++null
|
||||
console.log(4 || (null = 4));
|
||||
|
||||
1
test/input/invalid/assign_5.js
Normal file
1
test/input/invalid/assign_5.js
Normal file
@@ -0,0 +1 @@
|
||||
console.log(5 || ([]?.length ^= 5));
|
||||
1
test/input/invalid/for-await.js
Normal file
1
test/input/invalid/for-await.js
Normal file
@@ -0,0 +1 @@
|
||||
for await (; console.log(42););
|
||||
1
test/input/invalid/optional-template.js
Normal file
1
test/input/invalid/optional-template.js
Normal file
@@ -0,0 +1 @@
|
||||
console?.log``;
|
||||
@@ -1,11 +1,11 @@
|
||||
{
|
||||
"version": 3,
|
||||
"sources": [
|
||||
"input.js"
|
||||
],
|
||||
"names": [],
|
||||
"mappings": ";;;;;;;;;;;;;;eAAc,OAAO,CAAC,KAAD,C;IAAd,G,YAAA,G;;gBACS,OAAO,CAAC,OAAD,C;IAAhB,K,aAAA,K;;AAEP,GAAG,CAAC,CAAJ,OAAA,GAAG,qBAAM,GAAG,CAAC,CAAJ,CAAM,KAAK,CAAC,CAAZ,CAAN,EAAH",
|
||||
"sourcesContent": [
|
||||
"const {foo} = require(\"bar\");\nconst {hello} = require(\"world\");\n\nfoo.x(...foo.y(hello.z));\n"
|
||||
]
|
||||
}
|
||||
{
|
||||
"version": 3,
|
||||
"sources": [
|
||||
"input.js"
|
||||
],
|
||||
"names": [],
|
||||
"mappings": ";;;;;;;;;;;;;;eAAc,OAAO,CAAC,KAAD,C;IAAd,G,YAAA,G;;gBACS,OAAO,CAAC,OAAD,C;IAAhB,K,aAAA,K;;AAEP,GAAG,CAAC,CAAJ,OAAA,GAAG,qBAAM,GAAG,CAAC,CAAJ,CAAM,KAAK,CAAC,CAAZ,CAAN,EAAH",
|
||||
"sourcesContent": [
|
||||
"const {foo} = require(\"bar\");\nconst {hello} = require(\"world\");\n\nfoo.x(...foo.y(hello.z));\n"
|
||||
]
|
||||
}
|
||||
|
||||
@@ -1,8 +1,8 @@
|
||||
{
|
||||
"compress": false,
|
||||
"mangle": {
|
||||
"properties": {
|
||||
"regex": "/^_/"
|
||||
}
|
||||
}
|
||||
}
|
||||
{
|
||||
"compress": false,
|
||||
"mangle": {
|
||||
"properties": {
|
||||
"regex": "/^_/"
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@@ -58,22 +58,30 @@ if (typeof phantom == "undefined") {
|
||||
}).listen();
|
||||
server.on("listening", function() {
|
||||
var port = server.address().port;
|
||||
if (debug) {
|
||||
console.log("http://localhost:" + port + "/");
|
||||
} else (function install() {
|
||||
child_process.spawn(process.platform == "win32" ? "npm.cmd" : "npm", [
|
||||
if (debug) return console.log("http://localhost:" + port + "/");
|
||||
var cmd = process.platform == "win32" ? "npm.cmd" : "npm";
|
||||
|
||||
function npm(args, done) {
|
||||
child_process.spawn(cmd, args, { stdio: [ "ignore", 1, 2 ] }).on("exit", done);
|
||||
}
|
||||
|
||||
(function install() {
|
||||
npm([
|
||||
"install",
|
||||
"graceful-fs@4.2.6",
|
||||
"phantomjs-prebuilt@2.1.14",
|
||||
"--no-audit",
|
||||
"--no-optional",
|
||||
"--no-save",
|
||||
"--no-update-notifier",
|
||||
], {
|
||||
stdio: [ "ignore", 1, 2 ]
|
||||
}).on("exit", function(code) {
|
||||
], function(code) {
|
||||
if (code) {
|
||||
console.log("npm install failed with code", code);
|
||||
return install();
|
||||
return npm([
|
||||
"cache",
|
||||
"clean",
|
||||
"--force",
|
||||
], install);
|
||||
}
|
||||
var program = require("phantomjs-prebuilt").exec(process.argv[1], port);
|
||||
program.stdout.pipe(process.stdout);
|
||||
|
||||
@@ -29,7 +29,7 @@ describe("bin/uglifyjs", function() {
|
||||
var command = uglifyjscmd + ' test/input/comments/filter.js --comments all';
|
||||
exec(command, function(err, stdout) {
|
||||
if (err) throw err;
|
||||
assert.strictEqual(stdout, "// foo\n/*@preserve*/\n// bar\n\n");
|
||||
assert.strictEqual(stdout, "// foo\n/*@preserve*/\n// bar\n");
|
||||
done();
|
||||
});
|
||||
});
|
||||
@@ -37,7 +37,7 @@ describe("bin/uglifyjs", function() {
|
||||
var command = uglifyjscmd + ' test/input/comments/filter.js --comments /r/';
|
||||
exec(command, function(err, stdout) {
|
||||
if (err) throw err;
|
||||
assert.strictEqual(stdout, "/*@preserve*/\n// bar\n\n");
|
||||
assert.strictEqual(stdout, "/*@preserve*/\n// bar\n");
|
||||
done();
|
||||
});
|
||||
});
|
||||
@@ -45,7 +45,7 @@ describe("bin/uglifyjs", function() {
|
||||
var command = uglifyjscmd + ' test/input/comments/filter.js --comments';
|
||||
exec(command, function(err, stdout) {
|
||||
if (err) throw err;
|
||||
assert.strictEqual(stdout, "/*@preserve*/\n\n");
|
||||
assert.strictEqual(stdout, "/*@preserve*/\n");
|
||||
done();
|
||||
});
|
||||
});
|
||||
@@ -427,16 +427,30 @@ describe("bin/uglifyjs", function() {
|
||||
done();
|
||||
});
|
||||
});
|
||||
it("Should throw syntax error (++null)", function(done) {
|
||||
it("Should throw syntax error (null = 4)", function(done) {
|
||||
var command = uglifyjscmd + " test/input/invalid/assign_4.js";
|
||||
exec(command, function(err, stdout, stderr) {
|
||||
assert.ok(err);
|
||||
assert.strictEqual(stdout, "");
|
||||
assert.strictEqual(stderr.split(/\n/).slice(0, 4).join("\n"), [
|
||||
"Parse error at test/input/invalid/assign_4.js:1,0",
|
||||
"++null",
|
||||
"^",
|
||||
"ERROR: Invalid use of ++ operator",
|
||||
"Parse error at test/input/invalid/assign_4.js:1,23",
|
||||
"console.log(4 || (null = 4));",
|
||||
" ^",
|
||||
"ERROR: Invalid assignment",
|
||||
].join("\n"));
|
||||
done();
|
||||
});
|
||||
});
|
||||
it("Should throw syntax error ([]?.length ^= 5)", function(done) {
|
||||
var command = uglifyjscmd + " test/input/invalid/assign_5.js";
|
||||
exec(command, function(err, stdout, stderr) {
|
||||
assert.ok(err);
|
||||
assert.strictEqual(stdout, "");
|
||||
assert.strictEqual(stderr.split(/\n/).slice(0, 4).join("\n"), [
|
||||
"Parse error at test/input/invalid/assign_5.js:1,29",
|
||||
"console.log(5 || ([]?.length ^= 5));",
|
||||
" ^",
|
||||
"ERROR: Invalid assignment",
|
||||
].join("\n"));
|
||||
done();
|
||||
});
|
||||
@@ -679,6 +693,20 @@ describe("bin/uglifyjs", function() {
|
||||
done();
|
||||
});
|
||||
});
|
||||
it("Should throw syntax error (for-await)", function(done) {
|
||||
var command = uglifyjscmd + " test/input/invalid/for-await.js";
|
||||
exec(command, function(err, stdout, stderr) {
|
||||
assert.ok(err);
|
||||
assert.strictEqual(stdout, "");
|
||||
assert.strictEqual(stderr.split(/\n/).slice(0, 4).join("\n"), [
|
||||
"Parse error at test/input/invalid/for-await.js:1,11",
|
||||
"for await (; console.log(42););",
|
||||
" ^",
|
||||
"ERROR: Unexpected token: punc «;»",
|
||||
].join("\n"));
|
||||
done();
|
||||
});
|
||||
});
|
||||
it("Should throw syntax error (switch defaults)", function(done) {
|
||||
var command = uglifyjscmd + " test/input/invalid/switch.js";
|
||||
exec(command, function(err, stdout, stderr) {
|
||||
@@ -693,6 +721,20 @@ describe("bin/uglifyjs", function() {
|
||||
done();
|
||||
});
|
||||
});
|
||||
it("Should throw syntax error (console?.log``)", function(done) {
|
||||
var command = uglifyjscmd + " test/input/invalid/optional-template.js";
|
||||
exec(command, function(err, stdout, stderr) {
|
||||
assert.ok(err);
|
||||
assert.strictEqual(stdout, "");
|
||||
assert.strictEqual(stderr.split(/\n/).slice(0, 4).join("\n"), [
|
||||
"Parse error at test/input/invalid/optional-template.js:1,12",
|
||||
"console?.log``;",
|
||||
" ^",
|
||||
"ERROR: Invalid template on optional chain",
|
||||
].join("\n"));
|
||||
done();
|
||||
});
|
||||
});
|
||||
it("Should handle literal string as source map input", function(done) {
|
||||
var command = [
|
||||
uglifyjscmd,
|
||||
|
||||
@@ -392,12 +392,12 @@ describe("comments", function() {
|
||||
describe("comment filters", function() {
|
||||
it("Should be able to filter comments by passing regexp", function() {
|
||||
var ast = UglifyJS.parse("/*!test1*/\n/*test2*/\n//!test3\n//test4\n<!--test5\n<!--!test6\n-->test7\n-->!test8");
|
||||
assert.strictEqual(ast.print_to_string({comments: /^!/}), "/*!test1*/\n//!test3\n//!test6\n//!test8\n");
|
||||
assert.strictEqual(ast.print_to_string({comments: /^!/}), "/*!test1*/\n//!test3\n//!test6\n//!test8");
|
||||
});
|
||||
|
||||
it("Should be able to filter comments with the 'all' option", function() {
|
||||
var ast = UglifyJS.parse("/*!test1*/\n/*test2*/\n//!test3\n//test4\n<!--test5\n<!--!test6\n-->test7\n-->!test8");
|
||||
assert.strictEqual(ast.print_to_string({comments: "all"}), "/*!test1*/\n/*test2*/\n//!test3\n//test4\n//test5\n//!test6\n//test7\n//!test8\n");
|
||||
assert.strictEqual(ast.print_to_string({comments: "all"}), "/*!test1*/\n/*test2*/\n//!test3\n//test4\n//test5\n//!test6\n//test7\n//!test8");
|
||||
});
|
||||
|
||||
it("Should be able to filter commments with the 'some' option", function() {
|
||||
@@ -410,13 +410,12 @@ describe("comments", function() {
|
||||
var f = function(node, comment) {
|
||||
return comment.value.length === 8;
|
||||
};
|
||||
|
||||
assert.strictEqual(ast.print_to_string({comments: f}), "/*TEST 123*/\n//8 chars.\n");
|
||||
assert.strictEqual(ast.print_to_string({comments: f}), "/*TEST 123*/\n//8 chars.");
|
||||
});
|
||||
|
||||
it("Should be able to filter comments by passing regex in string format", function() {
|
||||
var ast = UglifyJS.parse("/*!test1*/\n/*test2*/\n//!test3\n//test4\n<!--test5\n<!--!test6\n-->test7\n-->!test8");
|
||||
assert.strictEqual(ast.print_to_string({comments: "/^!/"}), "/*!test1*/\n//!test3\n//!test6\n//!test8\n");
|
||||
assert.strictEqual(ast.print_to_string({comments: "/^!/"}), "/*!test1*/\n//!test3\n//!test6\n//!test8");
|
||||
});
|
||||
|
||||
it("Should be able to get the comment and comment type when using a function", function() {
|
||||
@@ -424,14 +423,12 @@ describe("comments", function() {
|
||||
var f = function(node, comment) {
|
||||
return comment.type == "comment1" || comment.type == "comment3";
|
||||
};
|
||||
|
||||
assert.strictEqual(ast.print_to_string({comments: f}), "//!test3\n//test4\n//test5\n//!test6\n");
|
||||
assert.strictEqual(ast.print_to_string({comments: f}), "//!test3\n//test4\n//test5\n//!test6");
|
||||
});
|
||||
|
||||
it("Should be able to filter comments by passing a boolean", function() {
|
||||
var ast = UglifyJS.parse("/*!test1*/\n/*test2*/\n//!test3\n//test4\n<!--test5\n<!--!test6\n-->test7\n-->!test8");
|
||||
|
||||
assert.strictEqual(ast.print_to_string({comments: true}), "/*!test1*/\n/*test2*/\n//!test3\n//test4\n//test5\n//!test6\n//test7\n//!test8\n");
|
||||
assert.strictEqual(ast.print_to_string({comments: true}), "/*!test1*/\n/*test2*/\n//!test3\n//test4\n//test5\n//!test6\n//test7\n//!test8");
|
||||
assert.strictEqual(ast.print_to_string({comments: false}), "");
|
||||
});
|
||||
|
||||
@@ -439,10 +436,8 @@ describe("comments", function() {
|
||||
var ast = UglifyJS.parse("#!Random comment\n//test1\n/*test2*/");
|
||||
var f = function(node, comment) {
|
||||
assert.strictEqual(comment.type === "comment5", false);
|
||||
|
||||
return true;
|
||||
};
|
||||
|
||||
assert.strictEqual(ast.print_to_string({comments: f}), "#!Random comment\n//test1\n/*test2*/");
|
||||
});
|
||||
|
||||
@@ -453,9 +448,8 @@ describe("comments", function() {
|
||||
|
||||
it("Should have no problem on multiple calls", function() {
|
||||
const options = {
|
||||
comments: /ok/
|
||||
comments: /ok/,
|
||||
};
|
||||
|
||||
assert.strictEqual(UglifyJS.parse("/* ok */ function a(){}").print_to_string(options), "/* ok */function a(){}");
|
||||
assert.strictEqual(UglifyJS.parse("/* ok */ function a(){}").print_to_string(options), "/* ok */function a(){}");
|
||||
assert.strictEqual(UglifyJS.parse("/* ok */ function a(){}").print_to_string(options), "/* ok */function a(){}");
|
||||
@@ -463,14 +457,14 @@ describe("comments", function() {
|
||||
|
||||
it("Should handle shebang and preamble correctly", function() {
|
||||
var code = UglifyJS.minify("#!/usr/bin/node\nvar x = 10;", {
|
||||
output: { preamble: "/* Build */" }
|
||||
output: { preamble: "/* Build */" },
|
||||
}).code;
|
||||
assert.strictEqual(code, "#!/usr/bin/node\n/* Build */\nvar x=10;");
|
||||
});
|
||||
|
||||
it("Should handle preamble without shebang correctly", function() {
|
||||
var code = UglifyJS.minify("var x = 10;", {
|
||||
output: { preamble: "/* Build */" }
|
||||
output: { preamble: "/* Build */" },
|
||||
}).code;
|
||||
assert.strictEqual(code, "/* Build */\nvar x=10;");
|
||||
});
|
||||
|
||||
@@ -1,21 +0,0 @@
|
||||
var assert = require("assert");
|
||||
var UglifyJS = require("../..");
|
||||
|
||||
describe("ie8", function() {
|
||||
it("Should be able to minify() with undefined as catch parameter in a try...catch statement", function() {
|
||||
assert.strictEqual(
|
||||
UglifyJS.minify([
|
||||
"function a(b){",
|
||||
" try {",
|
||||
" throw 'Stuff';",
|
||||
" } catch (undefined) {",
|
||||
" console.log('caught: ' + undefined);",
|
||||
" }",
|
||||
" console.log('undefined is ' + undefined);",
|
||||
" return b === undefined;",
|
||||
"};",
|
||||
].join("\n")).code,
|
||||
'function a(o){try{throw"Stuff"}catch(o){console.log("caught: "+o)}return console.log("undefined is "+void 0),void 0===o}'
|
||||
);
|
||||
});
|
||||
});
|
||||
@@ -7,11 +7,11 @@ describe("let", function() {
|
||||
// Produce a lot of variables in a function and run it through mangle.
|
||||
var s = '"dddddeeeeelllllooooottttt"; function foo() {';
|
||||
for (var i = 0; i < 18000; i++) {
|
||||
s += "var v" + i + "=0;";
|
||||
s += "var v" + i + "=[];";
|
||||
}
|
||||
s += '}';
|
||||
var result = UglifyJS.minify(s, {
|
||||
compress: false
|
||||
compress: false,
|
||||
}).code;
|
||||
|
||||
// Verify that select keywords and reserved keywords not produced
|
||||
@@ -41,7 +41,7 @@ describe("let", function() {
|
||||
}
|
||||
var result = UglifyJS.minify(s, {
|
||||
compress: false,
|
||||
ie8: true,
|
||||
ie: true,
|
||||
mangle: {
|
||||
properties: true,
|
||||
}
|
||||
|
||||
@@ -235,8 +235,8 @@ describe("minify", function() {
|
||||
}
|
||||
});
|
||||
var code = result.code;
|
||||
assert.strictEqual(code, "var a=/* */function(){foo()}();");
|
||||
})
|
||||
assert.strictEqual(code, "var a=function(){foo()}();");
|
||||
});
|
||||
});
|
||||
|
||||
describe("JS_Parse_Error", function() {
|
||||
|
||||
@@ -183,6 +183,24 @@ describe("test/reduce.js", function() {
|
||||
"// }",
|
||||
].join("\n"));
|
||||
});
|
||||
it("Should reduce `for (const ... in ...)` without invalid intermediate AST", function() {
|
||||
if (semver.satisfies(process.version, "<4")) return;
|
||||
var code = [
|
||||
"var a = 0;",
|
||||
"",
|
||||
"for (const b in [ 1, 2, 3 ]) {",
|
||||
" a = +a + 1 - .2;",
|
||||
" console.log(a);",
|
||||
"}",
|
||||
].join("\n");
|
||||
var result = reduce_test(code, {
|
||||
compress: {
|
||||
unsafe_math: true,
|
||||
},
|
||||
});
|
||||
if (result.error) throw result.error;
|
||||
assert.deepEqual(result.warnings, []);
|
||||
});
|
||||
it("Should reduce infinite loops with reasonable performance", function() {
|
||||
if (semver.satisfies(process.version, "<=0.10")) return;
|
||||
this.timeout(120000);
|
||||
@@ -264,35 +282,40 @@ describe("test/reduce.js", function() {
|
||||
});
|
||||
if (result.error) throw result.error;
|
||||
assert.deepEqual(result.warnings, []);
|
||||
assert.strictEqual(result.code.replace(/function \(/g, "function("), (semver.satisfies(process.version, "<=0.10") ? [
|
||||
"// Can't reproduce test failure",
|
||||
"// minify options: {",
|
||||
'// "compress": false,',
|
||||
'// "mangle": false,',
|
||||
'// "output": {',
|
||||
'// "beautify": true',
|
||||
"// }",
|
||||
"// }",
|
||||
] : [
|
||||
[
|
||||
"try{",
|
||||
"null[function(){}]",
|
||||
"}catch(e){",
|
||||
"console.log(e)",
|
||||
"}",
|
||||
].join(""),
|
||||
"// output: TypeError: Cannot read property 'function(){}' of null",
|
||||
"// ",
|
||||
"// minify: TypeError: Cannot read property 'function() {}' of null",
|
||||
"// ",
|
||||
"// options: {",
|
||||
'// "compress": false,',
|
||||
'// "mangle": false,',
|
||||
'// "output": {',
|
||||
'// "beautify": true',
|
||||
"// }",
|
||||
"// }",
|
||||
]).join("\n"));
|
||||
if (semver.satisfies(process.version, "<=0.10")) {
|
||||
assert.strictEqual(result.code, [
|
||||
"// Can't reproduce test failure",
|
||||
"// minify options: {",
|
||||
'// "compress": false,',
|
||||
'// "mangle": false,',
|
||||
'// "output": {',
|
||||
'// "beautify": true',
|
||||
"// }",
|
||||
"// }",
|
||||
].join("\n"));
|
||||
} else {
|
||||
var message = result.code.split(/\n/, 3)[1].slice("// output: ".length);
|
||||
assert.strictEqual(result.code, [
|
||||
[
|
||||
"try{",
|
||||
"null[function(){}]",
|
||||
"}catch(e){",
|
||||
"console.log(e)",
|
||||
"}",
|
||||
].join(""),
|
||||
"// output: " + message,
|
||||
"// ",
|
||||
"// minify: " + message.replace("(){}", "() {}"),
|
||||
"// ",
|
||||
"// options: {",
|
||||
'// "compress": false,',
|
||||
'// "mangle": false,',
|
||||
'// "output": {',
|
||||
'// "beautify": true',
|
||||
"// }",
|
||||
"// }",
|
||||
].join("\n"));
|
||||
}
|
||||
});
|
||||
it("Should maintain block-scope for const/let", function() {
|
||||
if (semver.satisfies(process.version, "<4")) return;
|
||||
@@ -361,4 +384,39 @@ describe("test/reduce.js", function() {
|
||||
if (result.error) throw result.error;
|
||||
assert.strictEqual(result.code, read("test/input/reduce/destructured_catch.reduced.js"));
|
||||
});
|
||||
it("Should not enumerate `toString` over global context", function() {
|
||||
if (semver.satisfies(process.version, "<8")) return;
|
||||
var code = [
|
||||
"(async function() {});",
|
||||
"for (var k in this);",
|
||||
"console.log(k, 42 + this);",
|
||||
].join("\n");
|
||||
var result = reduce_test(code, {
|
||||
mangle: false,
|
||||
});
|
||||
if (result.error) throw result.error;
|
||||
assert.strictEqual(result.code, [
|
||||
"// Can't reproduce test failure",
|
||||
"// minify options: {",
|
||||
'// "mangle": false',
|
||||
"// }",
|
||||
].join("\n"));
|
||||
});
|
||||
it("Should reduce object with method syntax without invalid intermediate AST", function() {
|
||||
if (semver.satisfies(process.version, "<4")) return;
|
||||
var code = [
|
||||
"console.log({",
|
||||
" f() {",
|
||||
" return 1 - .8;",
|
||||
" },",
|
||||
"}.f());",
|
||||
].join("\n");
|
||||
var result = reduce_test(code, {
|
||||
compress: {
|
||||
unsafe_math: true,
|
||||
},
|
||||
});
|
||||
if (result.error) throw result.error;
|
||||
assert.deepEqual(result.warnings, []);
|
||||
});
|
||||
});
|
||||
|
||||
@@ -101,6 +101,19 @@ describe("sourcemaps", function() {
|
||||
var map = JSON.parse(result.map);
|
||||
assert.deepEqual(map.names, []);
|
||||
});
|
||||
it("Should mark class properties", function() {
|
||||
var result = UglifyJS.minify([
|
||||
"class A {",
|
||||
" static P = 42",
|
||||
" set #q(v) {}",
|
||||
"}",
|
||||
].join("\n"), {
|
||||
sourceMap: true,
|
||||
});
|
||||
if (result.error) throw result.error;
|
||||
assert.strictEqual(result.code, "class A{static P=42;set#q(s){}}");
|
||||
assert.strictEqual(result.map, '{"version":3,"sources":["0"],"names":["A","P","#q","v"],"mappings":"MAAMA,EACFC,SAAW,GACXC,MAAOC"}');
|
||||
});
|
||||
it("Should mark array/object literals", function() {
|
||||
var result = UglifyJS.minify([
|
||||
"var obj = {};",
|
||||
|
||||
@@ -5,23 +5,23 @@ var acorn = require("acorn");
|
||||
var ufuzz = require("./ufuzz");
|
||||
var UglifyJS = require("..");
|
||||
|
||||
function try_beautify(code) {
|
||||
var beautified = UglifyJS.minify(code, {
|
||||
function beautify(ast) {
|
||||
var beautified = UglifyJS.minify(ast, {
|
||||
compress: false,
|
||||
mangle: false,
|
||||
output: {
|
||||
beautify: true,
|
||||
braces: true
|
||||
}
|
||||
braces: true,
|
||||
},
|
||||
});
|
||||
if (beautified.error) return beautified;
|
||||
return UglifyJS.minify(beautified.code, {
|
||||
compress: false,
|
||||
mangle: false,
|
||||
output: {
|
||||
ast: true,
|
||||
},
|
||||
});
|
||||
if (beautified.error) {
|
||||
console.log("// !!! beautify failed !!!");
|
||||
console.log(beautified.error.stack);
|
||||
console.log(code);
|
||||
} else {
|
||||
console.log("// (beautified)");
|
||||
console.log(beautified.code);
|
||||
}
|
||||
}
|
||||
|
||||
function validate(ast) {
|
||||
@@ -35,16 +35,55 @@ function validate(ast) {
|
||||
return UglifyJS.minify(ast, {
|
||||
compress: false,
|
||||
mangle: false,
|
||||
output: {
|
||||
ast: true,
|
||||
},
|
||||
validate: true,
|
||||
});
|
||||
}
|
||||
|
||||
function test(original, estree, description) {
|
||||
var transformed = validate(UglifyJS.AST_Node.from_mozilla_ast(estree));
|
||||
if (transformed.error || original !== transformed.code) {
|
||||
function patch_import(code) {
|
||||
return code.replace(/\bimport\s*\{\s*\}\s*from\s*(['"])/g, "import$1")
|
||||
.replace(/\b(import\b.*?)\s*,\s*\{\s*\}\s*(from\s*['"])/g, "$1 $2");
|
||||
}
|
||||
|
||||
function equals(input, transformed) {
|
||||
if (input.code === transformed.code) return true;
|
||||
return patch_import(input.code) === patch_import(transformed.code);
|
||||
}
|
||||
|
||||
function test(input, to_moz, description, skip_on_error, beautified) {
|
||||
try {
|
||||
var ast = UglifyJS.AST_Node.from_mozilla_ast(to_moz(input));
|
||||
} catch (e) {
|
||||
if (skip_on_error) return true;
|
||||
console.log("//=============================================================");
|
||||
console.log("//", description, "failed... round", round);
|
||||
console.log(e);
|
||||
console.log("// original code");
|
||||
if (beautified === true) console.log("// (beautified)");
|
||||
console.log(input.code);
|
||||
return false;
|
||||
}
|
||||
var transformed = validate(ast);
|
||||
if (transformed.error || !equals(input, transformed)) {
|
||||
if (!beautified) {
|
||||
beautified = beautify(input.ast);
|
||||
if (!beautified.error) {
|
||||
beautified.raw = beautified.code;
|
||||
if (!test(beautified, to_moz, description, skip_on_error, true)) return false;
|
||||
}
|
||||
}
|
||||
console.log("//=============================================================");
|
||||
console.log("// !!!!!! Failed... round", round);
|
||||
console.log("// original code");
|
||||
try_beautify(original);
|
||||
if (beautified.error) {
|
||||
console.log("// !!! beautify failed !!!");
|
||||
console.log(beautified.error.stack);
|
||||
} else if (beautified === true) {
|
||||
console.log("// (beautified)");
|
||||
}
|
||||
console.log(input.raw);
|
||||
console.log();
|
||||
console.log();
|
||||
console.log("//-------------------------------------------------------------");
|
||||
@@ -52,7 +91,15 @@ function test(original, estree, description) {
|
||||
if (transformed.error) {
|
||||
console.log(transformed.error.stack);
|
||||
} else {
|
||||
try_beautify(transformed.code);
|
||||
beautified = beautify(transformed.ast);
|
||||
if (beautified.error) {
|
||||
console.log("// !!! beautify failed !!!");
|
||||
console.log(beautified.error.stack);
|
||||
console.log(transformed.code);
|
||||
} else {
|
||||
console.log("// (beautified)");
|
||||
console.log(beautified.code);
|
||||
}
|
||||
}
|
||||
console.log("!!!!!! Failed... round", round);
|
||||
return false;
|
||||
@@ -67,24 +114,33 @@ for (var round = 1; round <= num_iterations; round++) {
|
||||
process.stdout.write(round + " of " + num_iterations + "\r");
|
||||
var code = ufuzz.createTopLevelCode();
|
||||
minify_options.forEach(function(options) {
|
||||
var input = options ? UglifyJS.minify(code, JSON.parse(options)).code : code;
|
||||
var uglified = UglifyJS.minify(input, {
|
||||
var ok = true;
|
||||
var input = UglifyJS.minify(options ? UglifyJS.minify(code, JSON.parse(options)).code : code, {
|
||||
compress: false,
|
||||
mangle: false,
|
||||
output: {
|
||||
ast: true
|
||||
}
|
||||
ast: true,
|
||||
},
|
||||
});
|
||||
var ok = test(uglified.code, uglified.ast.to_mozilla_ast(), "AST_Node.to_mozilla_ast()");
|
||||
try {
|
||||
ok = test(uglified.code, acorn.parse(input), "acorn.parse()") && ok;
|
||||
} catch (e) {
|
||||
input.raw = options ? input.code : code;
|
||||
if (input.error) {
|
||||
ok = false;
|
||||
console.log("//=============================================================");
|
||||
console.log("// acorn parser failed... round", round);
|
||||
console.log(e);
|
||||
console.log("// minify() failed... round", round);
|
||||
console.log(input.error);
|
||||
console.log("// original code");
|
||||
console.log(input);
|
||||
console.log(code);
|
||||
}
|
||||
if (ok) ok = test(input, function(input) {
|
||||
return input.ast.to_mozilla_ast();
|
||||
}, "AST_Node.to_mozilla_ast()");
|
||||
if (ok) ok = test(input, function(input) {
|
||||
return acorn.parse(input.raw, {
|
||||
ecmaVersion: "latest",
|
||||
locations: true,
|
||||
sourceType: "module",
|
||||
});
|
||||
}, "acorn.parse()", !ufuzz.verbose);
|
||||
if (!ok) process.exit(1);
|
||||
});
|
||||
}
|
||||
|
||||
@@ -22,7 +22,7 @@ module.exports = function reduce_test(testcase, minify_options, reduce_options)
|
||||
reduce_options = reduce_options || {};
|
||||
var print_options = {};
|
||||
[
|
||||
"ie8",
|
||||
"ie",
|
||||
"v8",
|
||||
"webkit",
|
||||
].forEach(function(name) {
|
||||
@@ -46,7 +46,7 @@ module.exports = function reduce_test(testcase, minify_options, reduce_options)
|
||||
if (verbose) {
|
||||
log("// Node.js " + process.version + " on " + os.platform() + " " + os.arch());
|
||||
}
|
||||
if (differs.error && [ "DefaultsError", "SyntaxError" ].indexOf(differs.error.name) < 0) {
|
||||
if (differs && differs.error && [ "DefaultsError", "SyntaxError" ].indexOf(differs.error.name) < 0) {
|
||||
test_for_diff = test_minify;
|
||||
differs = test_for_diff(testcase, minify_options, result_cache, max_timeout);
|
||||
}
|
||||
@@ -132,6 +132,8 @@ module.exports = function reduce_test(testcase, minify_options, reduce_options)
|
||||
return;
|
||||
}
|
||||
if (parent instanceof U.AST_VarDef && parent.name === node) return;
|
||||
// preserve class methods
|
||||
if (parent instanceof U.AST_ClassMethod && parent.value === node) return;
|
||||
// preserve exports
|
||||
if (parent instanceof U.AST_ExportDeclaration) return;
|
||||
if (parent instanceof U.AST_ExportDefault) return;
|
||||
@@ -147,6 +149,9 @@ module.exports = function reduce_test(testcase, minify_options, reduce_options)
|
||||
if (parent instanceof U.AST_For && parent.init === node && node instanceof U.AST_Definitions) return node;
|
||||
// preserve for (xxx in/of ...)
|
||||
if (parent instanceof U.AST_ForEnumeration && parent.init === node) return node;
|
||||
// preserve super(...)
|
||||
if (node.TYPE == "Call" && node.expression instanceof U.AST_Super) return;
|
||||
if (node instanceof U.AST_Super && parent.TYPE == "Call" && parent.expression === node) return node;
|
||||
|
||||
// node specific permutations with no parent logic
|
||||
|
||||
@@ -208,12 +213,10 @@ module.exports = function reduce_test(testcase, minify_options, reduce_options)
|
||||
}
|
||||
if (node.expression instanceof U.AST_Function) {
|
||||
// hoist and return expressions from the IIFE function expression
|
||||
var body = node.expression.body;
|
||||
node.expression.body = [];
|
||||
var seq = [];
|
||||
body.forEach(function(node) {
|
||||
node.expression.body.forEach(function(node) {
|
||||
var expr = expr instanceof U.AST_Exit ? node.value : node.body;
|
||||
if (expr instanceof U.AST_Node && !U.is_statement(expr)) {
|
||||
if (expr instanceof U.AST_Node && !U.is_statement(expr) && can_hoist(expr)) {
|
||||
// collect expressions from each statements' body
|
||||
seq.push(expr);
|
||||
}
|
||||
@@ -264,7 +267,7 @@ module.exports = function reduce_test(testcase, minify_options, reduce_options)
|
||||
CHANGED = true;
|
||||
return List.skip;
|
||||
default:
|
||||
if (!has_exit(node)) {
|
||||
if (!has_exit(node) && can_hoist(node)) {
|
||||
// hoist function declaration body
|
||||
var body = node.body;
|
||||
node.body = [];
|
||||
@@ -318,10 +321,11 @@ module.exports = function reduce_test(testcase, minify_options, reduce_options)
|
||||
var expr;
|
||||
switch ((node.start._permute * steps | 0) % 3) {
|
||||
case 0:
|
||||
if (!(node.init instanceof U.AST_Definitions
|
||||
&& node.init.definitions[0].name instanceof U.AST_Destructured)) {
|
||||
expr = node.init;
|
||||
if (node.init instanceof U.AST_Definitions) {
|
||||
if (node.init instanceof U.AST_Const) break;
|
||||
if (node.init.definitions[0].name instanceof U.AST_Destructured) break;
|
||||
}
|
||||
expr = node.init;
|
||||
break;
|
||||
case 1:
|
||||
expr = node.object;
|
||||
@@ -381,11 +385,9 @@ module.exports = function reduce_test(testcase, minify_options, reduce_options)
|
||||
if (node.body instanceof U.AST_Call && node.body.expression instanceof U.AST_Function) {
|
||||
// hoist simple statement IIFE function expression body
|
||||
node.start._permute++;
|
||||
if (!has_exit(node.body.expression)) {
|
||||
var body = node.body.expression.body;
|
||||
node.body.expression.body = [];
|
||||
if (!has_exit(node.body.expression) && can_hoist(node.body.expression)) {
|
||||
CHANGED = true;
|
||||
return List.splice(body);
|
||||
return List.splice(node.body.expression.body);
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -488,23 +490,27 @@ module.exports = function reduce_test(testcase, minify_options, reduce_options)
|
||||
CHANGED = true;
|
||||
return newNode;
|
||||
}, function(node, in_list) {
|
||||
if (node instanceof U.AST_Sequence) {
|
||||
if (node instanceof U.AST_Definitions) {
|
||||
// remove empty var statement
|
||||
if (node.definitions.length == 0) return in_list ? List.skip : new U.AST_EmptyStatement({
|
||||
start: {},
|
||||
});
|
||||
} else if (node instanceof U.AST_ObjectMethod) {
|
||||
if (!/Function$/.test(node.value.TYPE)) return new U.AST_ObjectKeyVal({
|
||||
key: node.key,
|
||||
value: node.value,
|
||||
start: {},
|
||||
});
|
||||
} else if (node instanceof U.AST_Sequence) {
|
||||
// expand single-element sequence
|
||||
if (node.expressions.length == 1) return node.expressions[0];
|
||||
}
|
||||
else if (node instanceof U.AST_Try) {
|
||||
} else if (node instanceof U.AST_Try) {
|
||||
// expand orphaned try block
|
||||
if (!node.bcatch && !node.bfinally) return new U.AST_BlockStatement({
|
||||
body: node.body,
|
||||
start: {},
|
||||
});
|
||||
}
|
||||
else if (node instanceof U.AST_Definitions) {
|
||||
// remove empty var statement
|
||||
if (node.definitions.length == 0) return in_list ? List.skip : new U.AST_EmptyStatement({
|
||||
start: {},
|
||||
});
|
||||
}
|
||||
});
|
||||
|
||||
var diff_error_message;
|
||||
@@ -673,6 +679,20 @@ function has_loopcontrol(body, loop, label) {
|
||||
return found;
|
||||
}
|
||||
|
||||
function can_hoist(body) {
|
||||
var found = false;
|
||||
body.walk(new U.TreeWalker(function(node) {
|
||||
if (found) return true;
|
||||
if (node instanceof U.AST_NewTarget) return found = true;
|
||||
if (node instanceof U.AST_Scope) {
|
||||
if (node === body) return;
|
||||
return true;
|
||||
}
|
||||
if (node instanceof U.AST_Super) return found = true;
|
||||
}));
|
||||
return !found;
|
||||
}
|
||||
|
||||
function is_timed_out(result) {
|
||||
return sandbox.is_error(result) && /timed out/.test(result.message);
|
||||
}
|
||||
@@ -706,7 +726,7 @@ function to_statement_init(node) {
|
||||
return node instanceof U.AST_Const || node instanceof U.AST_Let ? new U.AST_BlockStatement({
|
||||
body: [ node ],
|
||||
start: {},
|
||||
}) : to_statement(node);;
|
||||
}) : to_statement(node);
|
||||
}
|
||||
|
||||
function wrap_with_console_log(node) {
|
||||
|
||||
@@ -4,16 +4,23 @@ alias uglify-js=$PWD/bin/uglifyjs
|
||||
UGLIFY_OPTIONS=$@
|
||||
|
||||
minify_in_situ() {
|
||||
ARGS="$UGLIFY_OPTIONS --in-situ"
|
||||
ARGS="$UGLIFY_OPTIONS --validate --in-situ"
|
||||
DIRS="$1"
|
||||
echo '> uglify-js' $DIRS $UGLIFY_OPTIONS
|
||||
for i in `find $DIRS -name '*.js'`
|
||||
for i in `find $DIRS -type f -name '*.js'`
|
||||
do
|
||||
ARGS="$ARGS $i"
|
||||
done
|
||||
uglify-js $ARGS
|
||||
}
|
||||
|
||||
npm_install() {
|
||||
PKG="$1"
|
||||
while !(npm install $PKG); do
|
||||
while !(npm cache clean --force); do echo "'npm cache clean' failed - retrying..."; done
|
||||
done
|
||||
}
|
||||
|
||||
rm -rf tmp/acorn \
|
||||
&& git clone https://github.com/acornjs/acorn.git tmp/acorn \
|
||||
&& cd tmp/acorn \
|
||||
@@ -89,7 +96,7 @@ minify_in_situ "acorn/src" \
|
||||
&& minify_in_situ "acorn-loose/src" \
|
||||
&& minify_in_situ "acorn-walk/src" \
|
||||
&& rm -rf node_modules \
|
||||
&& npm install \
|
||||
&& npm_install \
|
||||
&& rm -rf acorn/dist acorn-loose/dist acorn-walk/dist \
|
||||
&& npm run build \
|
||||
&& minify_in_situ "acorn/dist" \
|
||||
|
||||
189
test/release/bootstrap.sh
Executable file
189
test/release/bootstrap.sh
Executable file
@@ -0,0 +1,189 @@
|
||||
#!/bin/sh
|
||||
|
||||
alias uglify-js=$PWD/bin/uglifyjs
|
||||
UGLIFY_OPTIONS=$@
|
||||
|
||||
minify_in_situ() {
|
||||
ARGS="$UGLIFY_OPTIONS --validate --in-situ"
|
||||
DIRS="$1"
|
||||
echo '> uglify-js' $DIRS $UGLIFY_OPTIONS
|
||||
for i in `find $DIRS -type f -name '*.js'`
|
||||
do
|
||||
ARGS="$ARGS $i"
|
||||
done
|
||||
uglify-js $ARGS
|
||||
}
|
||||
|
||||
npm_install() {
|
||||
PKG="$1"
|
||||
while !(npm install $PKG); do
|
||||
while !(npm cache clean --force); do echo "'npm cache clean' failed - retrying..."; done
|
||||
done
|
||||
}
|
||||
|
||||
rm -rf tmp/bootstrap \
|
||||
&& git clone --depth 1 --branch v5.0.0-beta2 https://github.com/twbs/bootstrap.git tmp/bootstrap \
|
||||
&& cd tmp/bootstrap \
|
||||
&& rm -rf .git/hooks \
|
||||
&& patch -p1 <<EOF
|
||||
--- a/.babelrc.js
|
||||
+++ /dev/null
|
||||
@@ -1,12 +0,0 @@
|
||||
-module.exports = {
|
||||
- presets: [
|
||||
- [
|
||||
- '@babel/preset-env',
|
||||
- {
|
||||
- loose: true,
|
||||
- bugfixes: true,
|
||||
- modules: false
|
||||
- }
|
||||
- ]
|
||||
- ]
|
||||
-};
|
||||
--- a/.gitattributes
|
||||
+++ b/.gitattributes
|
||||
@@ -5,0 +6 @@
|
||||
+*.png binary
|
||||
--- a/build/build-plugins.js
|
||||
+++ b/build/build-plugins.js
|
||||
@@ -14 +13,0 @@ const rollup = require('rollup')
|
||||
-const { babel } = require('@rollup/plugin-babel')
|
||||
@@ -19,6 +17,0 @@ const plugins = [
|
||||
- babel({
|
||||
- // Only transpile our source code
|
||||
- exclude: 'node_modules/**',
|
||||
- // Include the helpers in each file, at most one copy of each
|
||||
- babelHelpers: 'bundled'
|
||||
- })
|
||||
--- a/build/rollup.config.js
|
||||
+++ b/build/rollup.config.js
|
||||
@@ -4 +3,0 @@ const path = require('path')
|
||||
-const { babel } = require('@rollup/plugin-babel')
|
||||
@@ -15,6 +13,0 @@ const plugins = [
|
||||
- babel({
|
||||
- // Only transpile our source code
|
||||
- exclude: 'node_modules/**',
|
||||
- // Include the helpers in the bundle, at most one copy of each
|
||||
- babelHelpers: 'bundled'
|
||||
- })
|
||||
--- a/js/tests/integration/rollup.bundle.js
|
||||
+++ b/js/tests/integration/rollup.bundle.js
|
||||
@@ -3 +2,0 @@
|
||||
-const { babel } = require('@rollup/plugin-babel')
|
||||
@@ -18,4 +16,0 @@ module.exports = {
|
||||
- babel({
|
||||
- exclude: 'node_modules/**',
|
||||
- babelHelpers: 'bundled'
|
||||
- })
|
||||
--- a/js/tests/karma.conf.js
|
||||
+++ b/js/tests/karma.conf.js
|
||||
@@ -7,2 +6,0 @@ const ip = require('ip')
|
||||
-const { babel } = require('@rollup/plugin-babel')
|
||||
-const istanbul = require('rollup-plugin-istanbul')
|
||||
@@ -84,13 +81,0 @@ const conf = {
|
||||
- istanbul({
|
||||
- exclude: [
|
||||
- 'node_modules/**',
|
||||
- 'js/tests/unit/**/*.spec.js',
|
||||
- 'js/tests/helpers/**/*.js'
|
||||
- ]
|
||||
- }),
|
||||
- babel({
|
||||
- // Only transpile our source code
|
||||
- exclude: 'node_modules/**',
|
||||
- // Inline the required helpers in each file
|
||||
- babelHelpers: 'inline'
|
||||
- }),
|
||||
@@ -142 +126,0 @@ if (BROWSERSTACK) {
|
||||
- 'karma-coverage-istanbul-reporter'
|
||||
@@ -144 +127,0 @@ if (BROWSERSTACK) {
|
||||
- reporters.push('coverage-istanbul')
|
||||
--- a/package.json
|
||||
+++ b/package.json
|
||||
@@ -23 +23 @@
|
||||
- "start": "npm-run-all --parallel watch docs-serve",
|
||||
+ "start": "npm-run-all --parallel watch",
|
||||
@@ -28,3 +27,0 @@
|
||||
- "css-lint": "npm-run-all --continue-on-error --parallel css-lint-*",
|
||||
- "css-lint-stylelint": "stylelint \"**/*.{css,scss}\" --cache --cache-location .cache/.stylelintcache --rd",
|
||||
- "css-lint-vars": "fusv scss/ site/assets/scss/",
|
||||
@@ -44 +40,0 @@
|
||||
- "js-lint": "eslint --cache --cache-location .cache/.eslintcache --report-unused-disable-directives .",
|
||||
@@ -46,3 +42,3 @@
|
||||
- "js-minify-standalone": "terser --compress passes=2 --mangle --comments \"/^!/\" --source-map \"content=dist/js/bootstrap.js.map,includeSources,url=bootstrap.min.js.map\" --output dist/js/bootstrap.min.js dist/js/bootstrap.js",
|
||||
- "js-minify-standalone-esm": "terser --compress passes=2 --mangle --comments \"/^!/\" --source-map \"content=dist/js/bootstrap.esm.js.map,includeSources,url=bootstrap.esm.min.js.map\" --output dist/js/bootstrap.esm.min.js dist/js/bootstrap.esm.js",
|
||||
- "js-minify-bundle": "terser --compress passes=2 --mangle --comments \"/^!/\" --source-map \"content=dist/js/bootstrap.bundle.js.map,includeSources,url=bootstrap.bundle.min.js.map\" --output dist/js/bootstrap.bundle.min.js dist/js/bootstrap.bundle.js",
|
||||
+ "js-minify-standalone": "../../bin/uglifyjs --compress passes=2 --mangle --comments \"/^!/\" --source-map \"content=dist/js/bootstrap.js.map,includeSources,url=bootstrap.min.js.map\" --output dist/js/bootstrap.min.js dist/js/bootstrap.js",
|
||||
+ "js-minify-standalone-esm": "../../bin/uglifyjs --compress passes=2 --mangle --comments \"/^!/\" --source-map \"content=dist/js/bootstrap.esm.js.map,includeSources,url=bootstrap.esm.min.js.map\" --output dist/js/bootstrap.esm.min.js dist/js/bootstrap.esm.js",
|
||||
+ "js-minify-bundle": "../../bin/uglifyjs --compress passes=2 --mangle --comments \"/^!/\" --source-map \"content=dist/js/bootstrap.bundle.js.map,includeSources,url=bootstrap.bundle.min.js.map\" --output dist/js/bootstrap.bundle.min.js dist/js/bootstrap.bundle.js",
|
||||
@@ -56,25 +52 @@
|
||||
- "lint": "npm-run-all --parallel js-lint css-lint lockfile-lint",
|
||||
- "docs": "npm-run-all docs-build docs-lint",
|
||||
- "docs-build": "hugo --cleanDestinationDir",
|
||||
- "docs-compile": "npm run docs-build",
|
||||
- "docs-linkinator": "linkinator _gh_pages --recurse --silent --skip \"^(?!http://localhost)\"",
|
||||
- "docs-vnu": "node build/vnu-jar.js",
|
||||
- "docs-lint": "npm-run-all --parallel docs-vnu docs-linkinator",
|
||||
- "docs-serve": "hugo server --port 9001 --disableFastRender",
|
||||
- "docs-serve-only": "npx sirv-cli _gh_pages --port 9001",
|
||||
- "lockfile-lint": "lockfile-lint --allowed-hosts npm --allowed-schemes https: --empty-hostname false --type npm --path package-lock.json",
|
||||
- "update-deps": "ncu -u -x karma-browserstack-launcher,terser && npm update && echo Manually update site/assets/js/vendor",
|
||||
- "release": "npm-run-all dist release-sri docs-build release-zip*",
|
||||
- "release-sri": "node build/generate-sri.js",
|
||||
- "release-version": "node build/change-version.js",
|
||||
- "release-zip": "cross-env-shell \"rm -rf bootstrap-\$npm_package_version-dist && cp -r dist/ bootstrap-\$npm_package_version-dist && zip -r9 bootstrap-\$npm_package_version-dist.zip bootstrap-\$npm_package_version-dist && rm -rf bootstrap-\$npm_package_version-dist\"",
|
||||
- "release-zip-examples": "node build/zip-examples.js",
|
||||
- "dist": "npm-run-all --parallel css js",
|
||||
- "test": "npm-run-all lint dist js-test docs-build docs-lint",
|
||||
- "netlify": "cross-env-shell HUGO_BASEURL=\$DEPLOY_PRIME_URL npm-run-all dist release-sri docs-build",
|
||||
- "watch": "npm-run-all --parallel watch-*",
|
||||
- "watch-css-main": "nodemon --watch scss/ --ext scss --exec \"npm-run-all css-lint css-compile css-prefix\"",
|
||||
- "watch-css-dist": "nodemon --watch dist/css/ --ext css --ignore \"dist/css/*.rtl.*\" --exec \"npm run css-rtl\"",
|
||||
- "watch-css-docs": "nodemon --watch site/assets/scss/ --ext scss --exec \"npm run css-lint\"",
|
||||
- "watch-js-main": "nodemon --watch js/src/ --ext js --exec \"npm-run-all js-lint js-compile\"",
|
||||
- "watch-js-docs": "nodemon --watch site/assets/js/ --ext js --exec \"npm run js-lint\""
|
||||
+ "dist": "npm run css && npm run js"
|
||||
@@ -103,3 +74,0 @@
|
||||
- "@babel/cli": "^7.12.13",
|
||||
- "@babel/core": "^7.12.13",
|
||||
- "@babel/preset-env": "^7.12.13",
|
||||
@@ -107 +75,0 @@
|
||||
- "@rollup/plugin-babel": "^5.2.3",
|
||||
@@ -115,4 +82,0 @@
|
||||
- "eslint": "^7.19.0",
|
||||
- "eslint-config-xo": "^0.34.0",
|
||||
- "eslint-plugin-import": "^2.22.1",
|
||||
- "eslint-plugin-unicorn": "^27.0.0",
|
||||
@@ -122 +85,0 @@
|
||||
- "hugo-bin": "^0.68.0",
|
||||
@@ -128 +90,0 @@
|
||||
- "karma-coverage-istanbul-reporter": "^3.0.3",
|
||||
@@ -134,2 +95,0 @@
|
||||
- "linkinator": "^2.13.4",
|
||||
- "lockfile-lint": "^4.3.7",
|
||||
@@ -141 +100,0 @@
|
||||
- "rollup-plugin-istanbul": "^3.0.0",
|
||||
@@ -144,5 +103 @@
|
||||
- "shelljs": "^0.8.4",
|
||||
- "stylelint": "^13.9.0",
|
||||
- "stylelint-config-twbs-bootstrap": "^2.1.0",
|
||||
- "terser": "5.1.0",
|
||||
- "vnu-jar": "21.2.5"
|
||||
+ "shelljs": "^0.8.4"
|
||||
@@ -155,3 +109,0 @@
|
||||
- "hugo-bin": {
|
||||
- "buildTags": "extended"
|
||||
- },
|
||||
EOF
|
||||
ERR=$?; if [ "$ERR" != "0" ]; then echo "Error: $ERR"; exit $ERR; fi
|
||||
rm -rf node_modules \
|
||||
&& npm_install \
|
||||
&& minify_in_situ "node_modules/@popperjs/core" \
|
||||
&& rm -rf dist/js/* \
|
||||
&& minify_in_situ "build" \
|
||||
&& minify_in_situ "js" \
|
||||
&& minify_in_situ "site" \
|
||||
&& npm run dist \
|
||||
&& minify_in_situ "dist" \
|
||||
&& npm run js-test
|
||||
@@ -4,16 +4,23 @@ alias uglify-js=$PWD/bin/uglifyjs
|
||||
UGLIFY_OPTIONS=$@
|
||||
|
||||
minify_in_situ() {
|
||||
ARGS="$UGLIFY_OPTIONS --in-situ"
|
||||
ARGS="$UGLIFY_OPTIONS --validate --in-situ"
|
||||
DIRS="$1"
|
||||
echo '> uglify-js' $DIRS $UGLIFY_OPTIONS
|
||||
for i in `find $DIRS -name '*.js'`
|
||||
for i in `find $DIRS -type f -name '*.js'`
|
||||
do
|
||||
ARGS="$ARGS $i"
|
||||
done
|
||||
uglify-js $ARGS
|
||||
}
|
||||
|
||||
npm_install() {
|
||||
PKG="$1"
|
||||
while !(npm install $PKG); do
|
||||
while !(npm cache clean --force); do echo "'npm cache clean' failed - retrying..."; done
|
||||
done
|
||||
}
|
||||
|
||||
rm -rf tmp/buble \
|
||||
&& git clone https://github.com/bublejs/buble.git tmp/buble \
|
||||
&& cd tmp/buble \
|
||||
@@ -38,7 +45,7 @@ EOF
|
||||
ERR=$?; if [ "$ERR" != "0" ]; then echo "Error: $ERR"; exit $ERR; fi
|
||||
minify_in_situ "src" \
|
||||
&& rm -rf node_modules \
|
||||
&& npm ci \
|
||||
&& npm_install \
|
||||
&& rm -rf dist \
|
||||
&& npm run build \
|
||||
&& minify_in_situ "dist" \
|
||||
|
||||
@@ -4,16 +4,23 @@ alias uglify-js=$PWD/bin/uglifyjs
|
||||
UGLIFY_OPTIONS=$@
|
||||
|
||||
minify_in_situ() {
|
||||
ARGS="$UGLIFY_OPTIONS --in-situ"
|
||||
ARGS="$UGLIFY_OPTIONS --validate --in-situ"
|
||||
DIRS="$1"
|
||||
echo '> uglify-js' $DIRS $UGLIFY_OPTIONS
|
||||
for i in `find $DIRS -name '*.js'`
|
||||
for i in `find $DIRS -type f -name '*.js'`
|
||||
do
|
||||
ARGS="$ARGS $i"
|
||||
done
|
||||
uglify-js $ARGS
|
||||
}
|
||||
|
||||
npm_install() {
|
||||
PKG="$1"
|
||||
while !(npm install $PKG); do
|
||||
while !(npm cache clean --force); do echo "'npm cache clean' failed - retrying..."; done
|
||||
done
|
||||
}
|
||||
|
||||
rm -rf tmp/butternut \
|
||||
&& git clone https://github.com/Rich-Harris/butternut.git tmp/butternut \
|
||||
&& cd tmp/butternut \
|
||||
@@ -38,7 +45,7 @@ EOF
|
||||
ERR=$?; if [ "$ERR" != "0" ]; then echo "Error: $ERR"; exit $ERR; fi
|
||||
minify_in_situ "src" \
|
||||
&& rm -rf node_modules \
|
||||
&& npm install \
|
||||
&& npm_install \
|
||||
&& rm -rf dist \
|
||||
&& npm run build \
|
||||
&& minify_in_situ "dist" \
|
||||
|
||||
58
test/release/install.sh
Executable file
58
test/release/install.sh
Executable file
@@ -0,0 +1,58 @@
|
||||
echo "::group::GitHub Environment Variables"
|
||||
echo "CI: $CI"
|
||||
echo "GITHUB_WORKFLOW: $GITHUB_WORKFLOW"
|
||||
echo "GITHUB_RUN_ID: $GITHUB_RUN_ID"
|
||||
echo "GITHUB_RUN_NUMBER: $GITHUB_RUN_NUMBER"
|
||||
echo "GITHUB_ACTION: $GITHUB_ACTION"
|
||||
echo "GITHUB_ACTIONS: $GITHUB_ACTIONS"
|
||||
echo "GITHUB_ACTOR: $GITHUB_ACTOR"
|
||||
echo "GITHUB_REPOSITORY: $GITHUB_REPOSITORY"
|
||||
echo "GITHUB_EVENT_NAME: $GITHUB_EVENT_NAME"
|
||||
echo "GITHUB_EVENT_PATH: $GITHUB_EVENT_PATH"
|
||||
echo "GITHUB_WORKSPACE: $GITHUB_WORKSPACE"
|
||||
echo "GITHUB_SHA: $GITHUB_SHA"
|
||||
echo "GITHUB_REF: $GITHUB_REF"
|
||||
echo "GITHUB_HEAD_REF: $GITHUB_HEAD_REF"
|
||||
echo "GITHUB_BASE_REF: $GITHUB_BASE_REF"
|
||||
echo "GITHUB_SERVER_URL: $GITHUB_SERVER_URL"
|
||||
echo "GITHUB_API_URL: $GITHUB_API_URL"
|
||||
echo "GITHUB_GRAPHQL_URL: $GITHUB_GRAPHQL_URL"
|
||||
echo "::endgroup::"
|
||||
|
||||
if command -v timeout &> /dev/null; then NATIVE=1; fi
|
||||
timeout() {
|
||||
T=$1
|
||||
shift
|
||||
shift
|
||||
shift
|
||||
expect <<EOF
|
||||
set timeout $T
|
||||
spawn -noecho sh -c "$@"
|
||||
expect timeout { exit 124 } eof
|
||||
catch wait ret
|
||||
exit [lindex \$ret 3]
|
||||
EOF
|
||||
return $?
|
||||
}
|
||||
if [ $NATIVE ]; then unset -f timeout; fi
|
||||
|
||||
while !(git clone --branch v1.6.0 --depth 1 https://github.com/jasongin/nvs.git ~/.nvs); do
|
||||
rm -rf ~/.nvs
|
||||
done
|
||||
while ! timeout 60 bash -c ". ~/.nvs/nvs.sh add $NODE && nvs use $NODE"; do
|
||||
cd ~/.nvs
|
||||
while !(git clean -xdf); do echo "'git clean' failed - retrying..."; done
|
||||
cd -
|
||||
done
|
||||
. ~/.nvs/nvs.sh --version
|
||||
nvs use $NODE
|
||||
node --version
|
||||
npm config set audit false
|
||||
npm config set optional false
|
||||
npm config set save false
|
||||
npm config set strict-ssl false
|
||||
npm config set update-notifier false
|
||||
npm --version
|
||||
while !(npm install); do
|
||||
while !(npm cache clean --force); do echo "'npm cache clean' failed - retrying..."; done
|
||||
done
|
||||
@@ -1,23 +1,30 @@
|
||||
#!/bin/sh
|
||||
|
||||
alias uglify-js=$PWD/bin/uglifyjs
|
||||
UGLIFY_OPTIONS=$@
|
||||
UGLIFY_OPTIONS="--annotations $@"
|
||||
|
||||
minify_in_situ() {
|
||||
ARGS="$UGLIFY_OPTIONS --in-situ"
|
||||
ARGS="$UGLIFY_OPTIONS --validate --in-situ"
|
||||
DIRS="$1"
|
||||
echo '> uglify-js' $DIRS $UGLIFY_OPTIONS
|
||||
for i in `find $DIRS -name '*.js'`
|
||||
for i in `find $DIRS -type f -name '*.js'`
|
||||
do
|
||||
ARGS="$ARGS $i"
|
||||
done
|
||||
for i in `find $DIRS -name '*.mjs'`
|
||||
for i in `find $DIRS -type f -name '*.mjs'`
|
||||
do
|
||||
ARGS="$ARGS $i"
|
||||
done
|
||||
uglify-js $ARGS
|
||||
}
|
||||
|
||||
npm_install() {
|
||||
PKG="$1"
|
||||
while !(npm install $PKG); do
|
||||
while !(npm cache clean --force); do echo "'npm cache clean' failed - retrying..."; done
|
||||
done
|
||||
}
|
||||
|
||||
rm -rf tmp/mathjs \
|
||||
&& git clone --depth 1 --branch v9.2.0 https://github.com/josdejong/mathjs.git tmp/mathjs \
|
||||
&& cd tmp/mathjs \
|
||||
@@ -46,11 +53,15 @@ rm -rf tmp/mathjs \
|
||||
@@ -68 +75 @@ export function format (value, options) {
|
||||
- return value.toString()
|
||||
+ return HACK(value).toString()
|
||||
--- a/test/node-tests/treeShaking/treeShaking.test.js
|
||||
+++ b/test/node-tests/treeShaking/treeShaking.test.js
|
||||
@@ -35 +35 @@ describe('tree shaking', function () {
|
||||
- it('should apply tree-shaking when bundling', function (done) {
|
||||
+ if (0) it('should apply tree-shaking when bundling', function (done) {
|
||||
--- a/test/node-tests/cli/cli.test.js
|
||||
+++ b/test/node-tests/cli/cli.test.js
|
||||
@@ -36 +35,0 @@ describe('command line interface', function () {
|
||||
- const path2 = path.join(__dirname, 'script2')
|
||||
@@ -38,2 +37,2 @@ describe('command line interface', function () {
|
||||
- run('"' + path1 + '" "' + path2 + '"', function (e, result) {
|
||||
- assert.strictEqual(result, '2\n8\n')
|
||||
+ run('"' + path1 + '"', function (e, result) {
|
||||
+ assert.strictEqual(result, '2\n')
|
||||
--- a/test/unit-tests/expression/node/Node.test.js
|
||||
+++ b/test/unit-tests/expression/node/Node.test.js
|
||||
@@ -157 +157 @@ describe('Node', function () {
|
||||
@@ -187,7 +198,7 @@ minify_in_situ "bin" \
|
||||
&& minify_in_situ "test" \
|
||||
&& minify_in_situ "tools" \
|
||||
&& rm -rf node_modules \
|
||||
&& npm ci \
|
||||
&& npm_install \
|
||||
&& rm -rf lib \
|
||||
&& npm run build \
|
||||
&& minify_in_situ "lib" \
|
||||
|
||||
@@ -4,16 +4,23 @@ alias uglify-js=$PWD/bin/uglifyjs
|
||||
UGLIFY_OPTIONS=$@
|
||||
|
||||
minify_in_situ() {
|
||||
ARGS="$UGLIFY_OPTIONS --in-situ"
|
||||
ARGS="$UGLIFY_OPTIONS --validate --in-situ"
|
||||
DIRS="$1"
|
||||
echo '> uglify-js' $DIRS $UGLIFY_OPTIONS
|
||||
for i in `find $DIRS -name '*.js'`
|
||||
for i in `find $DIRS -type f -name '*.js'`
|
||||
do
|
||||
ARGS="$ARGS $i"
|
||||
done
|
||||
uglify-js $ARGS
|
||||
}
|
||||
|
||||
npm_install() {
|
||||
PKG="$1"
|
||||
while !(npm install $PKG); do
|
||||
while !(npm cache clean --force); do echo "'npm cache clean' failed - retrying..."; done
|
||||
done
|
||||
}
|
||||
|
||||
rm -rf tmp/rollup \
|
||||
&& git clone https://github.com/rollup/rollup.git tmp/rollup \
|
||||
&& cd tmp/rollup \
|
||||
@@ -77,7 +84,7 @@ minify_in_situ "bin" \
|
||||
&& minify_in_situ "browser" \
|
||||
&& minify_in_situ "src" \
|
||||
&& rm -rf node_modules \
|
||||
&& npm ci \
|
||||
&& npm_install \
|
||||
&& rm -rf dist \
|
||||
&& npm run build \
|
||||
&& minify_in_situ "dist" \
|
||||
|
||||
@@ -4,19 +4,30 @@ alias uglify-js=$PWD/bin/uglifyjs
|
||||
UGLIFY_OPTIONS=$@
|
||||
|
||||
minify_in_situ() {
|
||||
ARGS="$UGLIFY_OPTIONS --in-situ"
|
||||
ARGS="$UGLIFY_OPTIONS --validate --in-situ"
|
||||
DIRS="$1"
|
||||
echo '> esbuild' $DIRS
|
||||
for i in `find $DIRS -type f -name '*.ts' | grep -v '\.d\.ts'`
|
||||
do
|
||||
echo "$i"
|
||||
CODE=`cat "$i"`
|
||||
node_modules/.bin/esbuild --loader=ts --target=esnext > "$i" <<EOF
|
||||
$CODE
|
||||
EOF
|
||||
ARGS="$ARGS $i"
|
||||
done
|
||||
echo '> uglify-js' $DIRS $UGLIFY_OPTIONS
|
||||
for i in `find $DIRS -name '*.js'`
|
||||
for i in `find $DIRS -type f -name '*.js'`
|
||||
do
|
||||
ARGS="$ARGS $i"
|
||||
done
|
||||
uglify-js $ARGS
|
||||
for i in `find $DIRS -name '*.ts' | grep -v '\.d\.ts'`
|
||||
do
|
||||
echo "$i"
|
||||
node_modules/.bin/esbuild --loader=ts --target=node14 < "$i" \
|
||||
| uglify-js $UGLIFY_OPTIONS -o "$i"
|
||||
}
|
||||
|
||||
npm_install() {
|
||||
PKG="$1"
|
||||
while !(npm install $PKG); do
|
||||
while !(npm cache clean --force); do echo "'npm cache clean' failed - retrying..."; done
|
||||
done
|
||||
}
|
||||
|
||||
@@ -32,13 +43,22 @@ rm -rf tmp/rollup \
|
||||
- "postpublish": "pinst --enable",
|
||||
- "prepare": "npm run build",
|
||||
- "prepublishOnly": "pinst --disable && npm ci && npm run lint:nofix && npm run security && npm run build:bootstrap && npm run test:all",
|
||||
@@ -93 +89 @@
|
||||
- "is-reference": "lukastaegert/is-reference#update-class-features",
|
||||
+ "is-reference": "3.0.0",
|
||||
--- a/test/cli/index.js
|
||||
+++ b/test/cli/index.js
|
||||
@@ -13,0 +14,3 @@ sander.rimrafSync(__dirname, 'node_modules');
|
||||
+sander.rimrafSync(__dirname, 'samples', 'watch', 'bundle-error');
|
||||
+sander.rimrafSync(__dirname, 'samples', 'watch', 'watch-config-error');
|
||||
+sander.rimrafSync(__dirname, 'samples', 'watch', 'watch-config-initial-error');
|
||||
EOF
|
||||
ERR=$?; if [ "$ERR" != "0" ]; then echo "Error: $ERR"; exit $ERR; fi
|
||||
npm install esbuild-wasm@0.8.56 \
|
||||
npm_install esbuild-wasm@0.8.56 \
|
||||
&& minify_in_situ "cli" \
|
||||
&& minify_in_situ "src" \
|
||||
&& rm -rf node_modules \
|
||||
&& npm ci \
|
||||
&& npm_install \
|
||||
&& rm -rf dist \
|
||||
&& npm run build \
|
||||
&& minify_in_situ "dist" \
|
||||
|
||||
@@ -4,23 +4,34 @@ alias uglify-js=$PWD/bin/uglifyjs
|
||||
UGLIFY_OPTIONS=$@
|
||||
|
||||
minify_in_situ() {
|
||||
ARGS="$UGLIFY_OPTIONS --in-situ"
|
||||
ARGS="$UGLIFY_OPTIONS --validate --in-situ"
|
||||
DIRS="$1"
|
||||
echo '> esbuild' $DIRS
|
||||
for i in `find $DIRS -type f -name '*.ts' | grep -v '\.d\.ts'`
|
||||
do
|
||||
echo "$i"
|
||||
CODE=`cat "$i"`
|
||||
node_modules/.bin/esbuild --loader=ts --target=esnext > "$i" <<EOF
|
||||
$CODE
|
||||
EOF
|
||||
ARGS="$ARGS $i"
|
||||
done
|
||||
echo '> uglify-js' $DIRS $UGLIFY_OPTIONS
|
||||
for i in `find $DIRS -name '*.js'`
|
||||
for i in `find $DIRS -type f -name '*.js'`
|
||||
do
|
||||
ARGS="$ARGS $i"
|
||||
done
|
||||
for i in `find $DIRS -name '*.mjs'`
|
||||
for i in `find $DIRS -type f -name '*.mjs'`
|
||||
do
|
||||
ARGS="$ARGS $i"
|
||||
done
|
||||
uglify-js $ARGS
|
||||
for i in `find $DIRS -name '*.ts' | grep -v '\.d\.ts'`
|
||||
do
|
||||
echo "$i"
|
||||
node_modules/.bin/esbuild --loader=ts --target=node14 < "$i" \
|
||||
| uglify-js $UGLIFY_OPTIONS -o "$i"
|
||||
}
|
||||
|
||||
npm_install() {
|
||||
PKG="$1"
|
||||
while !(npm install $PKG); do
|
||||
while !(npm cache clean --force); do echo "'npm cache clean' failed - retrying..."; done
|
||||
done
|
||||
}
|
||||
|
||||
@@ -79,10 +90,10 @@ rm -rf tmp/sucrase \
|
||||
+export { getJSXPragmaInfo as HACK };
|
||||
EOF
|
||||
ERR=$?; if [ "$ERR" != "0" ]; then echo "Error: $ERR"; exit $ERR; fi
|
||||
npm install esbuild-wasm@0.8.56 \
|
||||
npm_install esbuild-wasm@0.8.56 \
|
||||
&& minify_in_situ "src" \
|
||||
&& rm -rf node_modules \
|
||||
&& npm install \
|
||||
&& npm_install \
|
||||
&& npm run clean \
|
||||
&& npm run build \
|
||||
&& minify_in_situ "dist" \
|
||||
|
||||
65
test/release/web-tooling-benchmark.sh
Executable file
65
test/release/web-tooling-benchmark.sh
Executable file
@@ -0,0 +1,65 @@
|
||||
#!/bin/sh
|
||||
|
||||
alias uglify-js="node --max-old-space-size=8192 $PWD/bin/uglifyjs"
|
||||
UGLIFY_OPTIONS=$@
|
||||
|
||||
minify_in_situ() {
|
||||
ARGS="$UGLIFY_OPTIONS --validate --in-situ"
|
||||
DIRS="$1"
|
||||
echo '> uglify-js' $DIRS $UGLIFY_OPTIONS
|
||||
for i in `find $DIRS -type f -name '*.js'`
|
||||
do
|
||||
ARGS="$ARGS $i"
|
||||
done
|
||||
uglify-js $ARGS
|
||||
}
|
||||
|
||||
npm_install() {
|
||||
PKG="$1"
|
||||
while !(npm install $PKG); do
|
||||
while !(npm cache clean --force); do echo "'npm cache clean' failed - retrying..."; done
|
||||
done
|
||||
}
|
||||
|
||||
rm -rf tmp/web-tooling-benchmark \
|
||||
&& git clone --depth 1 --branch v0.5.3 https://github.com/v8/web-tooling-benchmark.git tmp/web-tooling-benchmark \
|
||||
&& cd tmp/web-tooling-benchmark \
|
||||
&& rm -rf .git/hooks \
|
||||
&& patch -l -p1 <<EOF
|
||||
--- a/package.json
|
||||
+++ b/package.json
|
||||
@@ -12 +11,0 @@
|
||||
- "postinstall": "npm run build:terser-bundled && npm run build:uglify-js-bundled && npm run build",
|
||||
--- a/src/bootstrap.js
|
||||
+++ b/src/bootstrap.js
|
||||
@@ -6 +6 @@ const gmean = require("compute-gmean");
|
||||
-const package = require("../package.json");
|
||||
+const package_json = require("../package.json");
|
||||
@@ -65 +65 @@ function initialize() {
|
||||
- document.title = \`Web Tooling Benchmark v\${package.version}\`;
|
||||
+ document.title = \`Web Tooling Benchmark v\${package_json.version}\`;
|
||||
@@ -68 +68 @@ function initialize() {
|
||||
- versionDiv.innerHTML = \`v\${package.version}\`;
|
||||
+ versionDiv.innerHTML = \`v\${package_json.version}\`;
|
||||
--- a/src/cli-flags-helper.js
|
||||
+++ b/src/cli-flags-helper.js
|
||||
@@ -7 +6,0 @@ const targetList = new Set([
|
||||
- "chai",
|
||||
--- a/src/cli.js
|
||||
+++ b/src/cli.js
|
||||
@@ -18,0 +19 @@ suite.on("error", event => {
|
||||
+ global.process.exitCode = 42;
|
||||
EOF
|
||||
ERR=$?; if [ "$ERR" != "0" ]; then echo "Error: $ERR"; exit $ERR; fi
|
||||
minify_in_situ "src" \
|
||||
&& minify_in_situ "third_party" \
|
||||
&& rm -rf node_modules \
|
||||
&& npm_install \
|
||||
&& rm -rf build/* \
|
||||
&& npm run build:terser-bundled \
|
||||
&& npm run build:uglify-js-bundled \
|
||||
&& minify_in_situ "build" \
|
||||
&& rm -rf dist \
|
||||
&& npm run build \
|
||||
&& minify_in_situ "dist" \
|
||||
&& node dist/cli.js
|
||||
@@ -27,8 +27,8 @@ exports.run_code = semver.satisfies(process.version, "0.8") ? function(code, top
|
||||
} : semver.satisfies(process.version, "<0.12") ? run_code_vm : function(code, toplevel, timeout) {
|
||||
if ([
|
||||
/\basync[ \t]*\([\s\S]*?\)[ \t]*=>/,
|
||||
/\b(async[ \t]+function|setImmediate|setInterval|setTimeout)\b/,
|
||||
/\basync([ \t]+|[ \t]*\*[ \t]*)[^\s()[\]{},.&|!~=*%/+-]+(\s*\(|[ \t]*=>)/,
|
||||
/\b(async[ \t]+function|Promise|setImmediate|setInterval|setTimeout)\b/,
|
||||
/\basync([ \t]+|[ \t]*#|[ \t]*\*[ \t]*)[^\s()[\]{},.&|!~=*%/+-]+(\s*\(|[ \t]*=>)/,
|
||||
].some(function(pattern) {
|
||||
return pattern.test(code);
|
||||
})) {
|
||||
@@ -55,6 +55,8 @@ exports.patch_module_statements = function(code) {
|
||||
if (!header) return "";
|
||||
if (header.length == 1) return "0, " + header;
|
||||
return header.slice(0, -1) + " _" + ++count + header.slice(-1);
|
||||
}).replace(/\bimport\.meta\b/g, function() {
|
||||
return '({ url: "https://example.com/path/index.html" })';
|
||||
}).replace(/\bimport\b(?:\s*([^('"]+)\bfrom\b)?\s*(['"]).*?\2(?:$|\n|;)/g, function(match, symbols) {
|
||||
if (symbols) {
|
||||
if (!/^[{*]/.test(symbols)) symbols = "default:" + symbols;
|
||||
@@ -200,9 +202,13 @@ function setup(global, builtins, setup_log, setup_tty) {
|
||||
});
|
||||
Object.defineProperties(global, props);
|
||||
// for Node.js v8+
|
||||
global.toString = function() {
|
||||
return "[object global]";
|
||||
};
|
||||
if (global.toString !== Object.prototype.toString) {
|
||||
global.__proto__ = Object.defineProperty(Object.create(global.__proto__), "toString", {
|
||||
value: function() {
|
||||
return "[object global]";
|
||||
},
|
||||
});
|
||||
}
|
||||
|
||||
function self() {
|
||||
return this;
|
||||
@@ -216,7 +222,7 @@ function setup(global, builtins, setup_log, setup_tty) {
|
||||
if (arg === global) return "[object global]";
|
||||
if (/Error$/.test(arg.name)) return arg.toString();
|
||||
if (typeof arg.then == "function") return "[object Promise]";
|
||||
arg.constructor.toString();
|
||||
if (arg.constructor) arg.constructor.toString();
|
||||
var index = cache.original.indexOf(arg);
|
||||
if (index >= 0) return cache.replaced[index];
|
||||
if (--cache.level < 0) return "[object Object]";
|
||||
|
||||
@@ -1,7 +1,7 @@
|
||||
var get = require("https").get;
|
||||
var parse = require("url").parse;
|
||||
|
||||
var base, token, run_number, eldest = true;
|
||||
var base, token, run_number;
|
||||
exports.init = function(url, auth, num) {
|
||||
base = url;
|
||||
token = auth;
|
||||
@@ -19,14 +19,14 @@ exports.should_stop = function(callback) {
|
||||
do {
|
||||
workflow = runs.pop();
|
||||
if (!workflow) return;
|
||||
if (workflow.event == "schedule" && workflow.run_number == run_number) found = true;
|
||||
if (is_cron(workflow) && workflow.run_number == run_number) found = true;
|
||||
} while (!found && workflow.status == "completed");
|
||||
read(workflow.jobs_url, function(reply) {
|
||||
if (!reply || !Array.isArray(reply.jobs)) return;
|
||||
if (!reply.jobs.every(function(job) {
|
||||
if (job.status == "completed") return true;
|
||||
remaining--;
|
||||
return found || workflow.event != "schedule";
|
||||
return found || !is_cron(workflow);
|
||||
})) return;
|
||||
if (remaining >= 0) {
|
||||
next();
|
||||
@@ -38,6 +38,10 @@ exports.should_stop = function(callback) {
|
||||
});
|
||||
};
|
||||
|
||||
function is_cron(workflow) {
|
||||
return /^(schedule|workflow_dispatch|workflow_run)$/.test(workflow.event);
|
||||
}
|
||||
|
||||
function read(url, callback) {
|
||||
var done = function(reply) {
|
||||
done = function() {};
|
||||
@@ -56,7 +60,7 @@ function read(url, callback) {
|
||||
}).on("end", function() {
|
||||
var reply;
|
||||
try {
|
||||
reply = JSON.parse(chunks.join(""))
|
||||
reply = JSON.parse(chunks.join(""));
|
||||
} catch (e) {}
|
||||
done(reply);
|
||||
}).on("error", function() {
|
||||
|
||||
@@ -149,7 +149,10 @@ var SUPPORT = function(matrix) {
|
||||
for_of: "for (var a of []);",
|
||||
generator: "function* f(){}",
|
||||
let: "let a;",
|
||||
logical_assignment: "[].p ??= 0;",
|
||||
new_target: "function f() { new.target; }",
|
||||
nullish: "0 ?? 0",
|
||||
optional_chaining: "0?.p",
|
||||
rest: "var [...a] = [];",
|
||||
rest_object: "var {...a} = {};",
|
||||
spread: "[...[]];",
|
||||
@@ -157,6 +160,9 @@ var SUPPORT = function(matrix) {
|
||||
template: "``",
|
||||
trailing_comma: "function f(a,) {}",
|
||||
});
|
||||
if (SUPPORT.exponentiation && sandbox.run_code("console.log(10 ** 100 === Math.pow(10, 100));") !== "true\n") {
|
||||
SUPPORT.exponentiation = false;
|
||||
}
|
||||
|
||||
var VALUES = [
|
||||
'"a"',
|
||||
@@ -199,12 +205,20 @@ var VALUES = [
|
||||
'"function"',
|
||||
"this",
|
||||
];
|
||||
VALUES = VALUES.concat(VALUES);
|
||||
VALUES = VALUES.concat(VALUES);
|
||||
VALUES = VALUES.concat(VALUES);
|
||||
if (SUPPORT.bigint) VALUES = VALUES.concat([
|
||||
"(!0o644n)",
|
||||
"([3n][0] > 2)",
|
||||
"(-42n).toString()",
|
||||
"Number(0XDEADn << 16n | 0xbeefn)",
|
||||
]);
|
||||
VALUES = VALUES.concat(VALUES);
|
||||
VALUES = VALUES.concat(VALUES);
|
||||
VALUES = VALUES.concat(VALUES);
|
||||
VALUES = VALUES.concat(VALUES);
|
||||
VALUES.push("import.meta");
|
||||
|
||||
var BINARY_OPS = [
|
||||
" + ", // spaces needed to disambiguate with ++ cases (could otherwise cause syntax errors)
|
||||
@@ -240,51 +254,31 @@ BINARY_OPS = BINARY_OPS.concat(BINARY_OPS);
|
||||
BINARY_OPS = BINARY_OPS.concat(BINARY_OPS);
|
||||
BINARY_OPS.push(" in ");
|
||||
|
||||
var ASSIGNMENTS = [
|
||||
"=",
|
||||
"=",
|
||||
"=",
|
||||
"=",
|
||||
"=",
|
||||
"=",
|
||||
"=",
|
||||
"=",
|
||||
"=",
|
||||
"=",
|
||||
|
||||
"=",
|
||||
"=",
|
||||
"=",
|
||||
"=",
|
||||
"=",
|
||||
"=",
|
||||
"=",
|
||||
"=",
|
||||
"=",
|
||||
"=",
|
||||
|
||||
"+=",
|
||||
"+=",
|
||||
"+=",
|
||||
"+=",
|
||||
"+=",
|
||||
"+=",
|
||||
"+=",
|
||||
"+=",
|
||||
"+=",
|
||||
"+=",
|
||||
|
||||
var ASSIGNMENTS = [ "=" ];
|
||||
ASSIGNMENTS = ASSIGNMENTS.concat(ASSIGNMENTS);
|
||||
ASSIGNMENTS.push("+=");
|
||||
ASSIGNMENTS = ASSIGNMENTS.concat(ASSIGNMENTS);
|
||||
ASSIGNMENTS = ASSIGNMENTS.concat(ASSIGNMENTS);
|
||||
ASSIGNMENTS = ASSIGNMENTS.concat(ASSIGNMENTS);
|
||||
ASSIGNMENTS = ASSIGNMENTS.concat([
|
||||
"-=",
|
||||
"*=",
|
||||
"/=",
|
||||
"%=",
|
||||
"&=",
|
||||
"|=",
|
||||
"^=",
|
||||
"<<=",
|
||||
">>=",
|
||||
">>>=",
|
||||
"%=",
|
||||
];
|
||||
]);
|
||||
ASSIGNMENTS = ASSIGNMENTS.concat(ASSIGNMENTS);
|
||||
if (SUPPORT.exponentiation) ASSIGNMENTS.push("**=");
|
||||
if (SUPPORT.logical_assignment) ASSIGNMENTS = ASSIGNMENTS.concat([
|
||||
"&&=",
|
||||
"||=",
|
||||
"??=",
|
||||
]);
|
||||
|
||||
var UNARY_SAFE = [
|
||||
"+",
|
||||
@@ -362,6 +356,7 @@ var block_vars = [];
|
||||
var lambda_vars = [];
|
||||
var unique_vars = [];
|
||||
var classes = [];
|
||||
var allow_this = true;
|
||||
var async = false;
|
||||
var has_await = false;
|
||||
var export_default = false;
|
||||
@@ -408,6 +403,7 @@ function createTopLevelCode() {
|
||||
lambda_vars.length = 0;
|
||||
unique_vars.length = 0;
|
||||
classes.length = 0;
|
||||
allow_this = true;
|
||||
async = false;
|
||||
has_await = false;
|
||||
export_default = false;
|
||||
@@ -452,9 +448,9 @@ function addTrailingComma(list) {
|
||||
|
||||
function createParams(was_async, was_generator, noDuplicate) {
|
||||
var save_async = async;
|
||||
if (was_async) async = true;
|
||||
if (!async) async = was_async;
|
||||
var save_generator = generator;
|
||||
if (was_generator) generator = true;
|
||||
if (!generator) generator = was_generator;
|
||||
var len = unique_vars.length;
|
||||
var params = [];
|
||||
for (var n = rng(4); --n >= 0;) {
|
||||
@@ -576,9 +572,9 @@ function createAssignmentPairs(recurmax, stmtDepth, canThrow, nameLenBefore, was
|
||||
function createName() {
|
||||
unique_vars.push("a", "b", "c", "undefined", "NaN", "Infinity");
|
||||
var save_async = async;
|
||||
if (was_async) async = true;
|
||||
if (!async) async = was_async;
|
||||
var save_generator = generator;
|
||||
if (was_generator) generator = true;
|
||||
if (!generator) generator = was_generator;
|
||||
var name = createVarName(MANDATORY);
|
||||
generator = save_generator;
|
||||
async = save_async;
|
||||
@@ -989,8 +985,17 @@ function createStatement(recurmax, canThrow, canBreak, canContinue, cannotReturn
|
||||
var label = createLabel(canBreak, canContinue);
|
||||
canBreak = label.break || enableLoopControl(canBreak, CAN_BREAK);
|
||||
canContinue = label.continue || enableLoopControl(canContinue, CAN_CONTINUE);
|
||||
var key = rng(10) ? "key" + loop : getVarName(NO_CONST);
|
||||
var of = SUPPORT.for_of && rng(20) == 0;
|
||||
var key;
|
||||
if (rng(10)) {
|
||||
key = "key" + loop;
|
||||
} else if (bug_for_of_async && of) {
|
||||
addAvoidVar("async");
|
||||
key = getVarName(NO_CONST);
|
||||
removeAvoidVar("async");
|
||||
} else {
|
||||
key = getVarName(NO_CONST);
|
||||
}
|
||||
var init = "";
|
||||
if (!/^key/.test(key)) {
|
||||
if (!(of && bug_for_of_var) && rng(10) == 0) init = "var ";
|
||||
@@ -1186,7 +1191,7 @@ function createStatement(recurmax, canThrow, canBreak, canContinue, cannotReturn
|
||||
|
||||
function createSwitchParts(recurmax, n, canThrow, canBreak, canContinue, cannotReturn, stmtDepth) {
|
||||
var hadDefault = false;
|
||||
var s = [""];
|
||||
var s = [ "" ];
|
||||
canBreak = enableLoopControl(canBreak, CAN_BREAK);
|
||||
while (n-- > 0) {
|
||||
//hadDefault = n > 0; // disables weird `default` clause positioning (use when handling destabilizes)
|
||||
@@ -1400,13 +1405,16 @@ function _createExpression(recurmax, noComma, stmtDepth, canThrow) {
|
||||
createBlockVariables(recurmax, stmtDepth, canThrow, function(defns) {
|
||||
s.push(
|
||||
instantiate + makeFunction(name) + "(" + createParams(save_async, save_generator) + "){",
|
||||
strictMode(),
|
||||
defns()
|
||||
strictMode()
|
||||
);
|
||||
var add_new_target = SUPPORT.new_target && VALUES.indexOf("new.target") < 0;
|
||||
if (add_new_target) VALUES.push("new.target");
|
||||
s.push(defns());
|
||||
if (instantiate) for (var i = rng(4); --i >= 0;) {
|
||||
s.push((in_class ? "if (this) " : "") + createThisAssignment(recurmax, stmtDepth, canThrow));
|
||||
}
|
||||
s.push(_createStatements(rng(5) + 1, recurmax, canThrow, CANNOT_BREAK, CANNOT_CONTINUE, CAN_RETURN, stmtDepth));
|
||||
if (add_new_target) VALUES.splice(VALUES.indexOf("new.target"), 1);
|
||||
});
|
||||
generator = save_generator;
|
||||
async = save_async;
|
||||
@@ -1482,17 +1490,23 @@ function _createExpression(recurmax, noComma, stmtDepth, canThrow) {
|
||||
return createValue() + " in " + createObjectLiteral(recurmax, stmtDepth, canThrow);
|
||||
case p++:
|
||||
var name = getVarName();
|
||||
var s = name + "[" + createExpression(recurmax, COMMA_OK, stmtDepth, canThrow) + "]";
|
||||
return canThrow && rng(20) == 0 ? s : name + " && " + s;
|
||||
var prop = "[" + createExpression(recurmax, COMMA_OK, stmtDepth, canThrow) + "]";
|
||||
if (SUPPORT.optional_chaining && rng(50) == 0) return name + "?." + prop;
|
||||
if (canThrow && rng(20) == 0) return name + prop;
|
||||
return name + " && " + name + prop;
|
||||
case p++:
|
||||
var name = getVarName();
|
||||
var s = name + "." + getDotKey();
|
||||
return canThrow && rng(20) == 0 ? s : name + " && " + s;
|
||||
var prop = getDotKey();
|
||||
if (SUPPORT.optional_chaining && rng(50) == 0) return name + "?." + prop;
|
||||
if (canThrow && rng(20) == 0) return name + "." + prop;
|
||||
return name + " && " + name + "." + prop;
|
||||
case p++:
|
||||
case p++:
|
||||
var name = getVarName();
|
||||
var fn = name + "." + getDotKey();
|
||||
var s = "typeof " + fn + ' == "function" && --_calls_ >= 0 && ' + fn + createArgs(recurmax, stmtDepth, canThrow);
|
||||
var s = "typeof " + fn + ' == "function" && --_calls_ >= 0 && ';
|
||||
s += rng(5) ? fn : "(" + createExpression(recurmax, NO_COMMA, stmtDepth, canThrow) + ", " + fn + ")";
|
||||
s += createArgs(recurmax, stmtDepth, canThrow);
|
||||
return mayDefer(canThrow && rng(20) == 0 ? s : name + " && " + s);
|
||||
case p++:
|
||||
if (SUPPORT.class) {
|
||||
@@ -1527,7 +1541,16 @@ function _createExpression(recurmax, noComma, stmtDepth, canThrow) {
|
||||
name = rng(3) == 0 ? getVarName() : "f" + rng(funcs + 2);
|
||||
} while (name in called && !called[name]);
|
||||
called[name] = true;
|
||||
return mayDefer("typeof " + name + ' == "function" && --_calls_ >= 0 && ' + name + createArgs(recurmax, stmtDepth, canThrow));
|
||||
var args = createArgs(recurmax, stmtDepth, canThrow);
|
||||
var call = "typeof " + name + ' == "function" && --_calls_ >= 0 && ' + name + args;
|
||||
if (canThrow) {
|
||||
if (SUPPORT.optional_chaining && args[0] != "`" && rng(50) == 0) {
|
||||
call = "--_calls_ >= 0 && " + name + "?." + args;
|
||||
} else if (rng(20) == 0) {
|
||||
call = "--_calls_ >= 0 && " + name + args;
|
||||
}
|
||||
}
|
||||
return mayDefer(call);
|
||||
}
|
||||
_createExpression.N = p;
|
||||
return _createExpression(recurmax, noComma, stmtDepth, canThrow);
|
||||
@@ -1612,6 +1635,10 @@ var KEYS = [
|
||||
"1.5",
|
||||
"3",
|
||||
].concat(SAFE_KEYS);
|
||||
SAFE_KEYS = SAFE_KEYS.concat(SAFE_KEYS);
|
||||
SAFE_KEYS = SAFE_KEYS.concat(SAFE_KEYS);
|
||||
SAFE_KEYS = SAFE_KEYS.concat(SAFE_KEYS);
|
||||
SAFE_KEYS.push("__proto__");
|
||||
|
||||
function getDotKey(assign) {
|
||||
var key;
|
||||
@@ -1706,6 +1733,8 @@ function createObjectFunction(recurmax, stmtDepth, canThrow, internal, isClazz)
|
||||
fn = function(defns) {
|
||||
if (generator) name = "*" + name;
|
||||
if (async) name = "async "+ name;
|
||||
var save_allow = allow_this;
|
||||
if (internal == "super") allow_this = false;
|
||||
s = [
|
||||
name + "(" + createParams(save_async, save_generator, NO_DUPLICATE) + "){",
|
||||
strictMode(),
|
||||
@@ -1713,6 +1742,7 @@ function createObjectFunction(recurmax, stmtDepth, canThrow, internal, isClazz)
|
||||
];
|
||||
s.push(_createStatements(2, recurmax, canThrow, CANNOT_BREAK, CANNOT_CONTINUE, CANNOT_RETURN, stmtDepth));
|
||||
if (internal == "super") s.push("super" + createArgs(recurmax, stmtDepth, canThrow, NO_TEMPLATE) + ";");
|
||||
allow_this = save_allow;
|
||||
if (/^(constructor|super)$/.test(internal) || rng(10) == 0) for (var i = rng(4); --i >= 0;) {
|
||||
s.push(rng(2) ? createSuperAssignment(recurmax, stmtDepth, canThrow) : createThisAssignment(recurmax, stmtDepth, canThrow));
|
||||
}
|
||||
@@ -1729,8 +1759,9 @@ function createObjectFunction(recurmax, stmtDepth, canThrow, internal, isClazz)
|
||||
|
||||
function createObjectLiteral(recurmax, stmtDepth, canThrow) {
|
||||
recurmax--;
|
||||
var obj = ["({"];
|
||||
var obj = [ "({" ];
|
||||
var offset = SUPPORT.spread_object ? 0 : SUPPORT.computed_key ? 2 : 4;
|
||||
var has_proto = false;
|
||||
for (var i = rng(6); --i >= 0;) switch (offset + rng(50 - offset)) {
|
||||
case 0:
|
||||
obj.push("..." + getVarName() + ",");
|
||||
@@ -1746,7 +1777,12 @@ function createObjectLiteral(recurmax, stmtDepth, canThrow) {
|
||||
obj.push(createObjectFunction(recurmax, stmtDepth, canThrow) + ",");
|
||||
break;
|
||||
default:
|
||||
obj.push(createObjectKey(recurmax, stmtDepth, canThrow) + ": " + createExpression(recurmax, COMMA_OK, stmtDepth, canThrow) + ",");
|
||||
if (has_proto || rng(200)) {
|
||||
obj.push(createObjectKey(recurmax, stmtDepth, canThrow) + ": " + createExpression(recurmax, COMMA_OK, stmtDepth, canThrow) + ",");
|
||||
} else {
|
||||
obj.push("__proto__: " + createExpression(recurmax, COMMA_OK, stmtDepth, canThrow) + " || {},");
|
||||
has_proto = true;
|
||||
}
|
||||
break;
|
||||
}
|
||||
obj.push("})");
|
||||
@@ -1789,7 +1825,7 @@ function createClassLiteral(recurmax, stmtDepth, canThrow, name) {
|
||||
if (SUPPORT.class_field && rng(2)) {
|
||||
s += internal || createObjectKey(recurmax, stmtDepth, canThrow);
|
||||
if (rng(5)) {
|
||||
async = false;
|
||||
async = bug_async_class_await && fixed && 0;
|
||||
generator = false;
|
||||
s += " = " + createExpression(recurmax, NO_COMMA, stmtDepth, fixed ? canThrow : CANNOT_THROW);
|
||||
generator = save_generator;
|
||||
@@ -1940,7 +1976,11 @@ function createTypeofExpr(recurmax, stmtDepth, canThrow) {
|
||||
}
|
||||
|
||||
function createValue() {
|
||||
return VALUES[rng(VALUES.length)];
|
||||
var v;
|
||||
do {
|
||||
v = VALUES[rng(VALUES.length)];
|
||||
} while (v == "new.target" && rng(200) || !allow_this && v == "this");
|
||||
return v;
|
||||
}
|
||||
|
||||
function createBinaryOp(noComma, canThrow) {
|
||||
@@ -1989,7 +2029,7 @@ function isBannedKeyword(name) {
|
||||
case "arguments":
|
||||
return in_class;
|
||||
case "await":
|
||||
return async;
|
||||
return async !== false;
|
||||
case "yield":
|
||||
return generator || in_class;
|
||||
}
|
||||
@@ -2034,6 +2074,7 @@ function createVarName(maybe, dontStore) {
|
||||
if (require.main !== module) {
|
||||
exports.createTopLevelCode = createTopLevelCode;
|
||||
exports.num_iterations = num_iterations;
|
||||
exports.verbose = verbose;
|
||||
return;
|
||||
}
|
||||
|
||||
@@ -2212,11 +2253,10 @@ function log(options) {
|
||||
function sort_globals(code) {
|
||||
var globals = run_code("throw Object.keys(this).sort(" + function(global) {
|
||||
return function(m, n) {
|
||||
return (n == "toString") - (m == "toString")
|
||||
|| (typeof global[n] == "function") - (typeof global[m] == "function")
|
||||
return (typeof global[n] == "function") - (typeof global[m] == "function")
|
||||
|| (m < n ? -1 : m > n ? 1 : 0);
|
||||
};
|
||||
} + "(this));" + code);
|
||||
} + "(this));\n" + code);
|
||||
if (!Array.isArray(globals)) {
|
||||
errorln();
|
||||
errorln();
|
||||
@@ -2254,6 +2294,20 @@ function fuzzy_match(original, uglified) {
|
||||
}
|
||||
}
|
||||
|
||||
function patch_proto() {
|
||||
[ Array, Boolean, Error, Function, Number, Object, RegExp, String ].forEach(function(type) {
|
||||
[ "toString", "valueOf" ].forEach(function(prop) {
|
||||
type.prototype[prop] = function(fn) {
|
||||
return function() {
|
||||
try {
|
||||
return fn.apply(this, arguments);
|
||||
} catch (e) {}
|
||||
};
|
||||
}(type.prototype[prop]);
|
||||
});
|
||||
});
|
||||
}
|
||||
|
||||
function is_error_timeout(ex) {
|
||||
return /timed out/.test(ex.message);
|
||||
}
|
||||
@@ -2263,7 +2317,7 @@ function is_error_in(ex) {
|
||||
}
|
||||
|
||||
function is_error_spread(ex) {
|
||||
return ex.name == "TypeError" && /Found non-callable @@iterator| is not iterable| is not a function/.test(ex.message);
|
||||
return ex.name == "TypeError" && /Found non-callable @@iterator| is not iterable| not a function/.test(ex.message);
|
||||
}
|
||||
|
||||
function is_error_recursion(ex) {
|
||||
@@ -2382,6 +2436,8 @@ if (SUPPORT.arrow && SUPPORT.async && SUPPORT.rest && typeof sandbox.run_code("a
|
||||
return ex.name == "SyntaxError" && ex.message == "Rest parameter must be last formal parameter";
|
||||
};
|
||||
}
|
||||
var bug_async_class_await = SUPPORT.async && SUPPORT.class_field && typeof sandbox.run_code("var await; async function f() { class A { static p = await; } }") != "string";
|
||||
var bug_for_of_async = SUPPORT.for_await_of && typeof sandbox.run_code("var async; for (async of []);") != "string";
|
||||
var bug_for_of_var = SUPPORT.for_of && SUPPORT.let && typeof sandbox.run_code("try {} catch (e) { for (var e of []); }") != "string";
|
||||
if (SUPPORT.destructuring && typeof sandbox.run_code("console.log([ 1 ], {} = 2);") != "string") {
|
||||
beautify_options.output.v8 = true;
|
||||
@@ -2458,8 +2514,8 @@ for (var round = 1; round <= num_iterations; round++) {
|
||||
if (!ok && errored && uglify_result.name == "ReferenceError" && original_result.name == "ReferenceError") ok = true;
|
||||
// ignore difference due to implicit strict-mode in `class`
|
||||
if (!ok && /\bclass\b/.test(original_code)) {
|
||||
var original_strict = run_code('"use strict";' + original_code, toplevel);
|
||||
var uglify_strict = run_code('"use strict";' + uglify_code, toplevel);
|
||||
var original_strict = run_code('"use strict";\n' + original_code, toplevel);
|
||||
var uglify_strict = run_code('"use strict";\n' + uglify_code, toplevel);
|
||||
if (typeof original_strict != "string") {
|
||||
ok = typeof uglify_strict != "string";
|
||||
} else {
|
||||
@@ -2470,6 +2526,12 @@ for (var round = 1; round <= num_iterations; round++) {
|
||||
if (!ok && errored && /\bimport\b/.test(original_code)) {
|
||||
if (is_error_redeclaration(uglify_result) && is_error_redeclaration(original_result)) ok = true;
|
||||
}
|
||||
// ignore difference due to `__proto__` assignment
|
||||
if (!ok && /\b__proto__\b/.test(original_code)) {
|
||||
var original_proto = run_code("(" + patch_proto + ")();\n" + original_code, toplevel);
|
||||
var uglify_proto = run_code("(" + patch_proto + ")();\n" + uglify_code, toplevel);
|
||||
ok = sandbox.same_stdout(original_proto, uglify_proto);
|
||||
}
|
||||
// ignore difference in error message caused by `in`
|
||||
if (!ok && errored && is_error_in(uglify_result) && is_error_in(original_result)) ok = true;
|
||||
// ignore difference in error message caused by spread syntax
|
||||
|
||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user