Compare commits
295 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
860aa9531b | ||
|
|
2547542873 | ||
|
|
3f8f0e246e | ||
|
|
12227ebbb0 | ||
|
|
1b4bd7082b | ||
|
|
0b6c185818 | ||
|
|
bfd0ac7f4b | ||
|
|
1a8f2ecc65 | ||
|
|
dc9e1ff0b1 | ||
|
|
ea10498902 | ||
|
|
69636dad69 | ||
|
|
3ee1b0d00d | ||
|
|
1e3ca4c6f7 | ||
|
|
839f4361f4 | ||
|
|
ae5c3ee8a1 | ||
|
|
77f7ae5ba2 | ||
|
|
2d0f8bcff5 | ||
|
|
f97e107c09 | ||
|
|
e9932e1314 | ||
|
|
eaa84d32df | ||
|
|
6e4aa0326f | ||
|
|
f9a4b36dd1 | ||
|
|
3acb5a329e | ||
|
|
6d94242318 | ||
|
|
bca83cb9df | ||
|
|
a841d45bc3 | ||
|
|
eb93d92357 | ||
|
|
a0250ec923 | ||
|
|
25801627be | ||
|
|
32ae994f88 | ||
|
|
03aec89f60 | ||
|
|
faf0190546 | ||
|
|
c8b0f685ee | ||
|
|
87b99162fb | ||
|
|
940887f20f | ||
|
|
0b2573c3fa | ||
|
|
157521066f | ||
|
|
f766babf5e | ||
|
|
436a29367c | ||
|
|
55418fd460 | ||
|
|
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 | ||
|
|
24b73a95fa | ||
|
|
862b1b77b5 | ||
|
|
b4944a31a4 | ||
|
|
58362d5ec7 | ||
|
|
01fa430a3e | ||
|
|
f4ee0f651c | ||
|
|
077512d151 | ||
|
|
e4848a7f5a | ||
|
|
f52b0e7c31 | ||
|
|
31e7d25cad | ||
|
|
12babdfe20 | ||
|
|
397e48b97e | ||
|
|
c7520b4b97 | ||
|
|
ad903e9240 | ||
|
|
83c3838b07 | ||
|
|
fa09f87589 | ||
|
|
2db1a141ab | ||
|
|
dd30ed6a9b | ||
|
|
cb50a2d192 | ||
|
|
20be5209c0 | ||
|
|
2a49760032 | ||
|
|
04ed818f0a | ||
|
|
10ca578ee5 | ||
|
|
955411e065 | ||
|
|
adcafce048 | ||
|
|
b1e05fd48a | ||
|
|
23b51287aa | ||
|
|
74dee5c445 | ||
|
|
ee27d87a08 | ||
|
|
62887f2c66 | ||
|
|
68b2dadc58 |
58
.github/workflows/build.yml
vendored
Normal file
58
.github/workflows/build.yml
vendored
Normal file
@@ -0,0 +1,58 @@
|
||||
name: Build testing
|
||||
on:
|
||||
pull_request:
|
||||
push:
|
||||
branches: [ master ]
|
||||
jobs:
|
||||
ufuzz:
|
||||
strategy:
|
||||
fail-fast: false
|
||||
matrix:
|
||||
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'
|
||||
script: butternut.sh
|
||||
- node: '14'
|
||||
script: mathjs.sh
|
||||
- node: '8'
|
||||
script: rollup-es.sh
|
||||
- node: '14'
|
||||
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:
|
||||
NODE: ${{ matrix.node }}
|
||||
OPTIONS: ${{ matrix.options }}
|
||||
SCRIPT: ${{ matrix.script }}
|
||||
steps:
|
||||
- uses: actions/checkout@v2
|
||||
- name: Perform uglify, build & test
|
||||
shell: bash
|
||||
run: |
|
||||
. ./test/release/install.sh
|
||||
./test/release/$SCRIPT $OPTIONS
|
||||
24
.github/workflows/ci.yml
vendored
24
.github/workflows/ci.yml
vendored
@@ -7,14 +7,9 @@ jobs:
|
||||
test:
|
||||
strategy:
|
||||
matrix:
|
||||
node: [ '0.8', '0.10', '0.12', '4', '6', '8', '10', '12', latest ]
|
||||
node: [ '0.10', '0.12', '4', '6', '8', '10', '12', '14', '16', latest ]
|
||||
os: [ ubuntu-latest, windows-latest ]
|
||||
script: [ compress, mocha, release/benchmark, release/jetstream ]
|
||||
exclude:
|
||||
- node: '0.8'
|
||||
script: release/benchmark
|
||||
- node: '0.8'
|
||||
script: release/jetstream
|
||||
name: ${{ matrix.node }} ${{ matrix.os }} ${{ matrix.script }}
|
||||
runs-on: ${{ matrix.os }}
|
||||
env:
|
||||
@@ -29,20 +24,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
|
||||
|
||||
334
README.md
334
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.
|
||||
|
||||
@@ -1141,18 +1154,14 @@ in total it's a bit more than just using UglifyJS's own parser.
|
||||
It's not well known, but whitespace removal and symbol mangling accounts
|
||||
for 95% of the size reduction in minified code for most JavaScript - not
|
||||
elaborate code transforms. One can simply disable `compress` to speed up
|
||||
Uglify builds by 3 to 4 times. In this fast `mangle`-only mode Uglify has
|
||||
comparable minify speeds and gzip sizes to
|
||||
[`butternut`](https://www.npmjs.com/package/butternut):
|
||||
Uglify builds by 3 to 5 times.
|
||||
|
||||
| d3.js | minify size | gzip size | minify time (seconds) |
|
||||
| --- | ---: | ---: | ---: |
|
||||
| original | 451,131 | 108,733 | - |
|
||||
| uglify-js@3.0.24 mangle=false, compress=false | 316,600 | 85,245 | 0.70 |
|
||||
| uglify-js@3.0.24 mangle=true, compress=false | 220,216 | 72,730 | 1.13 |
|
||||
| butternut@0.4.6 | 217,568 | 72,738 | 1.41 |
|
||||
| uglify-js@3.0.24 mangle=true, compress=true | 212,511 | 71,560 | 3.36 |
|
||||
| babili@0.1.4 | 210,713 | 72,140 | 12.64 |
|
||||
| original | 511,371 | 119,932 | - |
|
||||
| uglify-js@3.13.0 mangle=false, compress=false | 363,988 | 95,695 | 0.56 |
|
||||
| uglify-js@3.13.0 mangle=true, compress=false | 253,305 | 81,281 | 0.99 |
|
||||
| uglify-js@3.13.0 mangle=true, compress=true | 244,436 | 79,854 | 5.30 |
|
||||
|
||||
To enable fast minify mode from the CLI use:
|
||||
```
|
||||
@@ -1190,6 +1199,17 @@ To allow for better optimizations, the compiler makes various assumptions:
|
||||
- Object properties can be added, removed and modified (not prevented with
|
||||
`Object.defineProperty()`, `Object.defineProperties()`, `Object.freeze()`,
|
||||
`Object.preventExtensions()` or `Object.seal()`).
|
||||
- If array destructuring is present, index-like properties in `Array.prototype`
|
||||
have not been overridden:
|
||||
```javascript
|
||||
Object.prototype[0] = 42;
|
||||
var [ a ] = [];
|
||||
var { 0: b } = {};
|
||||
// 42 undefined
|
||||
console.log([][0], a);
|
||||
// 42 42
|
||||
console.log({}[0], b);
|
||||
```
|
||||
- Earlier versions of JavaScript will throw `SyntaxError` with the following:
|
||||
```javascript
|
||||
({
|
||||
@@ -1285,3 +1305,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.
|
||||
|
||||
93
bin/uglifyjs
93
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,11 +110,15 @@ 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.",
|
||||
" --webkit Support non-standard Safari/Webkit.",
|
||||
" --wrap <name> Embed everything as a function with “exports” corresponding to “name” globally.",
|
||||
"",
|
||||
"(internal debug use only)",
|
||||
" --in-situ Warning: replaces original source files with minified output.",
|
||||
" --reduce-test Reduce a standalone test case (assumes cloned repository).",
|
||||
].join("\n"));
|
||||
}
|
||||
@@ -139,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;
|
||||
@@ -194,6 +206,7 @@ function process_option(name, no_value) {
|
||||
case "no-rename":
|
||||
options.rename = false;
|
||||
break;
|
||||
case "in-situ":
|
||||
case "reduce-test":
|
||||
case "self":
|
||||
break;
|
||||
@@ -225,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");
|
||||
@@ -254,7 +268,19 @@ if (specified["self"]) {
|
||||
if (!options.wrap) options.wrap = "UglifyJS";
|
||||
paths = UglifyJS.FILES;
|
||||
}
|
||||
if (paths.length) {
|
||||
if (specified["in-situ"]) {
|
||||
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");
|
||||
files = {};
|
||||
files[convert_path(name)] = read_file(name);
|
||||
output = name;
|
||||
run();
|
||||
});
|
||||
} else if (paths.length) {
|
||||
simple_glob(paths).forEach(function(name) {
|
||||
files[convert_path(name)] = read_file(name);
|
||||
});
|
||||
@@ -291,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]);
|
||||
@@ -389,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);
|
||||
|
||||
106
lib/ast.js
106
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;
|
||||
}
|
||||
|
||||
@@ -139,7 +136,7 @@ var AST_Node = DEFNODE("Node", "start end", {
|
||||
}, null);
|
||||
|
||||
(AST_Node.log_function = function(fn, verbose) {
|
||||
if (!fn) {
|
||||
if (typeof fn != "function") {
|
||||
AST_Node.info = AST_Node.warn = noop;
|
||||
return;
|
||||
}
|
||||
@@ -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;
|
||||
@@ -874,6 +871,7 @@ var AST_ClassMethod = DEFNODE("ClassMethod", null, {
|
||||
$documentation: "A `class` method",
|
||||
_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");
|
||||
},
|
||||
}, AST_ClassProperty);
|
||||
@@ -1291,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;
|
||||
@@ -1313,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", {
|
||||
@@ -1335,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() {
|
||||
@@ -1683,6 +1689,8 @@ var AST_ObjectMethod = DEFNODE("ObjectMethod", null, {
|
||||
$documentation: "A key(){} object property",
|
||||
_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 object method's lambda must be null");
|
||||
},
|
||||
}, AST_ObjectKeyVal);
|
||||
|
||||
@@ -1753,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",
|
||||
@@ -1809,10 +1817,20 @@ var AST_Super = DEFNODE("Super", null, {
|
||||
var AST_This = DEFNODE("This", null, {
|
||||
$documentation: "The `this` symbol",
|
||||
_validate: function() {
|
||||
if (this.name !== "this") throw new Error('name must be "this"');
|
||||
if (this.TYPE == "This" && this.name !== "this") throw new Error('name must be "this"');
|
||||
},
|
||||
}, AST_ObjectIdentity);
|
||||
|
||||
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);
|
||||
},
|
||||
}, AST_This);
|
||||
|
||||
var AST_Template = DEFNODE("Template", "expressions strings tag", {
|
||||
$documentation: "A template literal, i.e. tag`str1${expr1}...strN${exprN}strN+1`",
|
||||
$propdoc: {
|
||||
@@ -1981,7 +1999,7 @@ TreeWalker.prototype = {
|
||||
},
|
||||
find_parent: function(type) {
|
||||
var stack = this.stack;
|
||||
for (var i = stack.length; --i >= 0;) {
|
||||
for (var i = stack.length - 1; --i >= 0;) {
|
||||
var x = stack[i];
|
||||
if (x instanceof type) return x;
|
||||
}
|
||||
|
||||
3929
lib/compress.js
3929
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
535
lib/output.js
535
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) {
|
||||
@@ -1133,8 +1152,9 @@ function OutputStream(options) {
|
||||
});
|
||||
}
|
||||
function print_arrow(self, output) {
|
||||
if (self.argnames.length == 1 && self.argnames[0] instanceof AST_SymbolFunarg && !self.rest) {
|
||||
self.argnames[0].print(output);
|
||||
var argname = self.argnames.length == 1 && !self.rest && self.argnames[0];
|
||||
if (argname instanceof AST_SymbolFunarg && argname.name != "yield") {
|
||||
argname.print(output);
|
||||
} else {
|
||||
print_funargs(self, output);
|
||||
}
|
||||
@@ -1265,7 +1285,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 +1315,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 +1456,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 +1479,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 +1510,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 +1620,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 +1685,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 +1726,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 +1772,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 +1828,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 +1880,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 +1932,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);
|
||||
});
|
||||
})();
|
||||
|
||||
366
lib/parse.js
366
lib/parse.js
@@ -44,10 +44,10 @@
|
||||
|
||||
"use strict";
|
||||
|
||||
var KEYWORDS = "break case catch class const continue debugger default delete do else extends finally for function if in instanceof let new return switch throw try typeof var void while with";
|
||||
var KEYWORDS = "break case catch class const continue debugger default delete do else extends finally for function if in instanceof new return switch throw try typeof var void while with";
|
||||
var KEYWORDS_ATOM = "false null true";
|
||||
var RESERVED_WORDS = [
|
||||
"abstract async await boolean byte char double enum export final float goto implements import int interface long native package private protected public short static super synchronized this throws transient volatile yield",
|
||||
"abstract async await boolean byte char double enum export final float goto implements import int interface let long native package private protected public short static super synchronized this throws transient volatile yield",
|
||||
KEYWORDS_ATOM,
|
||||
KEYWORDS,
|
||||
].join(" ");
|
||||
@@ -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;
|
||||
}
|
||||
@@ -238,10 +242,10 @@ function tokenizer($TEXT, filename, html5_comments, shebang) {
|
||||
read_template : with_eof_error("Unterminated template literal", function(strings) {
|
||||
var s = "";
|
||||
for (;;) {
|
||||
var ch = next(true, true);
|
||||
var ch = read();
|
||||
switch (ch) {
|
||||
case "\\":
|
||||
ch += next(true, true);
|
||||
ch += read();
|
||||
break;
|
||||
case "`":
|
||||
strings.push(s);
|
||||
@@ -256,6 +260,11 @@ function tokenizer($TEXT, filename, html5_comments, shebang) {
|
||||
}
|
||||
s += ch;
|
||||
}
|
||||
|
||||
function read() {
|
||||
var ch = next(true, true);
|
||||
return ch == "\r" ? "\n" : ch;
|
||||
}
|
||||
}),
|
||||
};
|
||||
var prev_was_dot = false;
|
||||
@@ -347,7 +356,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 +364,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 +473,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 +488,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 +623,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 +664,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,10 +862,20 @@ 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_();
|
||||
}
|
||||
break;
|
||||
case "let":
|
||||
if (is_vardefs()) {
|
||||
next();
|
||||
var node = let_();
|
||||
semicolon();
|
||||
return node;
|
||||
}
|
||||
break;
|
||||
case "yield":
|
||||
if (S.in_generator) return simple_statement();
|
||||
break;
|
||||
@@ -939,12 +961,6 @@ function parse($TEXT, options) {
|
||||
next();
|
||||
return if_();
|
||||
|
||||
case "let":
|
||||
next();
|
||||
var node = let_();
|
||||
semicolon();
|
||||
return node;
|
||||
|
||||
case "return":
|
||||
if (S.in_function == 0 && !options.bare_returns)
|
||||
croak("'return' outside of function");
|
||||
@@ -1053,6 +1069,16 @@ function parse($TEXT, options) {
|
||||
return stat;
|
||||
}
|
||||
|
||||
function has_modifier(name) {
|
||||
if (!is("name", name)) return;
|
||||
var token = peek();
|
||||
if (!token) return;
|
||||
if (is_token(token, "operator", "=")) return;
|
||||
if (token.type == "punc" && /^[(;}]$/.test(token.value)) return;
|
||||
if (has_newline_before(token)) return;
|
||||
return next();
|
||||
}
|
||||
|
||||
function class_(ctor) {
|
||||
var was_async = S.in_async;
|
||||
var was_gen = S.in_generator;
|
||||
@@ -1078,16 +1104,8 @@ function parse($TEXT, options) {
|
||||
continue;
|
||||
}
|
||||
var start = S.token;
|
||||
var fixed = is("name", "static");
|
||||
if (fixed) next();
|
||||
var async = is("name", "async") && peek();
|
||||
if (async) {
|
||||
if (async.type == "punc" && /^[(=;}]$/.test(async.value) || has_newline_before(async)) {
|
||||
async = false;
|
||||
} else {
|
||||
async = next();
|
||||
}
|
||||
}
|
||||
var fixed = !!has_modifier("static");
|
||||
var async = has_modifier("async");
|
||||
if (is("operator", "*")) {
|
||||
next();
|
||||
var internal = is("name") && /^#/.test(S.token.value);
|
||||
@@ -1179,10 +1197,10 @@ 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")
|
||||
: is("name", "let") && is_vardefs()
|
||||
? (next(), let_(true))
|
||||
: is("keyword", "var")
|
||||
? (next(), var_(true))
|
||||
@@ -1230,6 +1248,7 @@ function parse($TEXT, options) {
|
||||
}
|
||||
|
||||
function for_enum(ctor, init) {
|
||||
handle_regexp();
|
||||
var obj = expression();
|
||||
expect(")");
|
||||
return new ctor({
|
||||
@@ -1297,6 +1316,11 @@ function parse($TEXT, options) {
|
||||
}
|
||||
if (node instanceof AST_SymbolFunarg) return node;
|
||||
if (node instanceof AST_SymbolRef) return new AST_SymbolFunarg(node);
|
||||
if (node instanceof AST_Yield) return new AST_SymbolFunarg({
|
||||
start: node.start,
|
||||
name: "yield",
|
||||
end: node.end,
|
||||
});
|
||||
token_error(node.start, "Invalid arrow parameter");
|
||||
}
|
||||
|
||||
@@ -1323,21 +1347,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,
|
||||
@@ -1345,6 +1367,8 @@ function parse($TEXT, options) {
|
||||
value: value,
|
||||
end: prev(),
|
||||
});
|
||||
if (is_strict) node.each_argname(strict_verify_symbol);
|
||||
return node;
|
||||
}
|
||||
|
||||
var function_ = function(ctor) {
|
||||
@@ -1377,23 +1401,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_() {
|
||||
@@ -1485,6 +1510,7 @@ function parse($TEXT, options) {
|
||||
body.start = start;
|
||||
body.end = prev();
|
||||
} else {
|
||||
handle_regexp();
|
||||
body = expression();
|
||||
semicolon();
|
||||
}
|
||||
@@ -1502,14 +1528,14 @@ function parse($TEXT, options) {
|
||||
}
|
||||
|
||||
function export_default_decl() {
|
||||
switch (S.token.value) {
|
||||
case "async":
|
||||
if (is("name", "async")) {
|
||||
if (!is_token(peek(), "keyword", "function")) return;
|
||||
next();
|
||||
next();
|
||||
if (!is("operator", "*")) return maybe_named(AST_AsyncDefun, function_(AST_AsyncFunction));
|
||||
next();
|
||||
return maybe_named(AST_AsyncGeneratorDefun, function_(AST_AsyncGeneratorFunction));
|
||||
} else if (is("keyword")) switch (S.token.value) {
|
||||
case "class":
|
||||
next();
|
||||
return maybe_named(AST_DefClass, class_(AST_ClassExpression));
|
||||
@@ -1522,13 +1548,19 @@ function parse($TEXT, options) {
|
||||
}
|
||||
|
||||
var export_decl = embed_tokens(function() {
|
||||
switch (S.token.value) {
|
||||
if (is("name")) switch (S.token.value) {
|
||||
case "async":
|
||||
next();
|
||||
expect_token("keyword", "function");
|
||||
if (!is("operator", "*")) return function_(AST_AsyncDefun);
|
||||
next();
|
||||
return function_(AST_AsyncGeneratorDefun);
|
||||
case "let":
|
||||
next();
|
||||
var node = let_();
|
||||
semicolon();
|
||||
return node;
|
||||
} else if (is("keyword")) switch (S.token.value) {
|
||||
case "class":
|
||||
next();
|
||||
return class_(AST_DefClass);
|
||||
@@ -1542,11 +1574,6 @@ function parse($TEXT, options) {
|
||||
if (!is("operator", "*")) return function_(AST_Defun);
|
||||
next();
|
||||
return function_(AST_GeneratorDefun);
|
||||
case "let":
|
||||
next();
|
||||
var node = let_();
|
||||
semicolon();
|
||||
return node;
|
||||
case "var":
|
||||
next();
|
||||
var node = var_();
|
||||
@@ -1707,6 +1734,11 @@ function parse($TEXT, options) {
|
||||
return a;
|
||||
}
|
||||
|
||||
function is_vardefs() {
|
||||
var token = peek();
|
||||
return is_token(token, "name") || is_token(token, "punc", "[") || is_token(token, "punc", "{");
|
||||
}
|
||||
|
||||
var const_ = function(no_in) {
|
||||
return new AST_Const({
|
||||
start : prev(),
|
||||
@@ -1734,58 +1766,66 @@ function parse($TEXT, options) {
|
||||
var new_ = function(allow_calls) {
|
||||
var start = S.token;
|
||||
expect_token("operator", "new");
|
||||
var newexp = expr_atom(false), args;
|
||||
if (is("punc", "(")) {
|
||||
var call;
|
||||
if (is("punc", ".") && is_token(peek(), "name", "target")) {
|
||||
next();
|
||||
args = expr_list(")", !options.strict);
|
||||
next();
|
||||
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;
|
||||
}
|
||||
|
||||
@@ -1797,10 +1837,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", ")")) {
|
||||
@@ -1825,11 +1862,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 "[":
|
||||
@@ -2110,7 +2149,7 @@ function parse($TEXT, options) {
|
||||
}
|
||||
|
||||
function strict_verify_symbol(sym) {
|
||||
if (sym.name == "arguments" || sym.name == "eval")
|
||||
if (sym.name == "arguments" || sym.name == "eval" || sym.name == "let")
|
||||
token_error(sym.start, "Unexpected " + sym.name + " in strict mode");
|
||||
}
|
||||
|
||||
@@ -2206,20 +2245,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 = [];
|
||||
@@ -2230,53 +2257,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;
|
||||
@@ -2385,7 +2433,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) {
|
||||
@@ -2507,7 +2555,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;
|
||||
|
||||
65
lib/scope.js
65
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
|
||||
@@ -198,20 +201,23 @@ AST_Toplevel.DEFMETHOD("figure_out_scope", function(options) {
|
||||
} else if (node instanceof AST_SymbolConst) {
|
||||
var def = scope.def_variable(node);
|
||||
def.defun = defun;
|
||||
def.exported = exported;
|
||||
if (exported) def.exported = true;
|
||||
} else if (node instanceof AST_SymbolDefun) {
|
||||
defun.def_function(node, tw.parent()).exported = exported;
|
||||
var def = defun.def_function(node, tw.parent());
|
||||
if (exported) def.exported = true;
|
||||
entangle(defun, scope);
|
||||
} else if (node instanceof AST_SymbolFunarg) {
|
||||
defun.def_variable(node);
|
||||
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) {
|
||||
scope.def_variable(node).exported = exported;
|
||||
var def = scope.def_variable(node);
|
||||
if (exported) def.exported = true;
|
||||
} else if (node instanceof AST_SymbolVar) {
|
||||
defun.def_variable(node, node instanceof AST_SymbolImport ? undefined : null).exported = exported;
|
||||
var def = defun.def_variable(node, node instanceof AST_SymbolImport ? undefined : null);
|
||||
if (exported) def.exported = true;
|
||||
entangle(defun, scope);
|
||||
}
|
||||
|
||||
@@ -268,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;
|
||||
@@ -343,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) {
|
||||
@@ -389,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;
|
||||
}
|
||||
});
|
||||
@@ -435,6 +444,7 @@ AST_Scope.DEFMETHOD("init_vars", function(parent_scope) {
|
||||
});
|
||||
AST_Arrow.DEFMETHOD("init_vars", function(parent_scope) {
|
||||
init_scope_vars(this, parent_scope);
|
||||
return this;
|
||||
});
|
||||
AST_AsyncArrow.DEFMETHOD("init_vars", function(parent_scope) {
|
||||
init_scope_vars(this, parent_scope);
|
||||
@@ -562,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,
|
||||
@@ -792,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) {
|
||||
@@ -803,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;
|
||||
|
||||
@@ -143,7 +143,7 @@ function push_uniq(array, el) {
|
||||
}
|
||||
|
||||
function string_template(text, props) {
|
||||
return text.replace(/\{([^}]+)\}/g, function(str, p) {
|
||||
return text.replace(/\{([^{}]+)\}/g, function(str, p) {
|
||||
var value = props[p];
|
||||
return value instanceof AST_Node ? value.print_to_string() : value;
|
||||
});
|
||||
|
||||
@@ -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.0",
|
||||
"version": "3.14.4",
|
||||
"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",
|
||||
|
||||
@@ -10,33 +10,38 @@ var sandbox = require("./sandbox");
|
||||
var semver = require("semver");
|
||||
var U = require("./node");
|
||||
|
||||
var file = process.argv[2];
|
||||
var batch = 50;
|
||||
var dir = path.resolve(path.dirname(module.filename), "compress");
|
||||
if (file) {
|
||||
if (process.argv.length > 3) {
|
||||
var file = process.argv[2];
|
||||
var start = process.argv[3] | 0;
|
||||
var minify_options = require("./ufuzz/options.json").map(JSON.stringify);
|
||||
log("--- {file}", { file: file });
|
||||
var tests = parse_test(path.resolve(dir, file));
|
||||
process.exit(Object.keys(tests).filter(function(name) {
|
||||
process.exit(Object.keys(tests).slice(start, start + batch).filter(function(name) {
|
||||
return !test_case(tests[name]);
|
||||
}).length);
|
||||
} else {
|
||||
var files = fs.readdirSync(dir).filter(function(name) {
|
||||
var files = process.argv.length == 3 ? [ process.argv[2] ] : fs.readdirSync(dir).filter(function(name) {
|
||||
return /\.js$/i.test(name);
|
||||
});
|
||||
var failures = 0;
|
||||
var failed_files = Object.create(null);
|
||||
(function next() {
|
||||
var file = files.shift();
|
||||
if (file) {
|
||||
child_process.spawn(process.argv[0], [ process.argv[1], file ], {
|
||||
(function next(file, start, length) {
|
||||
if (start < length) {
|
||||
child_process.spawn(process.argv[0], [ process.argv[1], file, start, batch ], {
|
||||
stdio: [ "ignore", 1, 2 ]
|
||||
}).on("exit", function(code) {
|
||||
if (code) {
|
||||
failures += code;
|
||||
failed_files[file] = code;
|
||||
failed_files[file] = true;
|
||||
}
|
||||
next();
|
||||
next(file, start + batch, length);
|
||||
});
|
||||
} else if (file = files.shift()) {
|
||||
log("--- {file}", { file: file });
|
||||
start = 0;
|
||||
length = Object.keys(parse_test(path.resolve(dir, file))).length;
|
||||
next(file, start, length);
|
||||
} else if (failures) {
|
||||
console.error();
|
||||
console.error("!!! Failed " + failures + " test case(s).");
|
||||
@@ -257,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,292 @@ 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"
|
||||
}
|
||||
|
||||
evaluate_lazy_assignment: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
reduce_vars: true,
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var a = 42;
|
||||
console.log(a &&= "PASS");
|
||||
}
|
||||
expect: {
|
||||
console.log("PASS");
|
||||
}
|
||||
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"
|
||||
}
|
||||
|
||||
File diff suppressed because it is too large
Load Diff
@@ -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"
|
||||
}
|
||||
|
||||
File diff suppressed because it is too large
Load Diff
@@ -958,8 +958,7 @@ collapse_vars_misc: {
|
||||
}
|
||||
expect: {
|
||||
function f0(o, a, h) {
|
||||
var b = 3 - a;
|
||||
return o.run(b)[7] = h;
|
||||
return o.run(3 - a)[7] = h;
|
||||
}
|
||||
function f1(x) { return 5 - x }
|
||||
function f2(x) { return foo() / (5 - x) }
|
||||
@@ -2265,6 +2264,7 @@ var_defs: {
|
||||
properties: true,
|
||||
sequences: true,
|
||||
side_effects: true,
|
||||
unsafe: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
@@ -2276,8 +2276,8 @@ var_defs: {
|
||||
}
|
||||
expect: {
|
||||
var f1 = function(x, y) {
|
||||
var r = x + y, z = r * r - r, b = 7;
|
||||
console.log(z + b);
|
||||
var r = x + y;
|
||||
console.log(r * r - r + 7);
|
||||
};
|
||||
f1("1", 0);
|
||||
}
|
||||
@@ -2487,6 +2487,7 @@ issue_1858: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
pure_getters: true,
|
||||
reduce_vars: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
@@ -2869,7 +2870,7 @@ lvalues_def: {
|
||||
expect_stdout: true
|
||||
}
|
||||
|
||||
compound_assignment: {
|
||||
compound_assignment_1: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
}
|
||||
@@ -2888,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: {
|
||||
@@ -2907,8 +2926,7 @@ issue_2187_1: {
|
||||
var a = 1;
|
||||
!function(foo) {
|
||||
foo();
|
||||
var a = 2;
|
||||
console.log(a);
|
||||
console.log(2);
|
||||
}(function() {
|
||||
console.log(a);
|
||||
});
|
||||
@@ -4256,6 +4274,7 @@ issue_2436_14: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
reduce_vars: true,
|
||||
unsafe: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
@@ -4746,6 +4765,7 @@ unsafe_builtin: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
pure_getters: "strict",
|
||||
reduce_vars: true,
|
||||
unsafe: true,
|
||||
unused: true,
|
||||
}
|
||||
@@ -5988,7 +6008,7 @@ issue_3215_1: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
evaluate: true,
|
||||
ie8: false,
|
||||
ie: false,
|
||||
inline: true,
|
||||
passes: 2,
|
||||
side_effects: true,
|
||||
@@ -6010,7 +6030,7 @@ issue_3215_2: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
evaluate: true,
|
||||
ie8: true,
|
||||
ie: true,
|
||||
inline: true,
|
||||
passes: 2,
|
||||
side_effects: true,
|
||||
@@ -6035,7 +6055,7 @@ issue_3215_3: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
evaluate: true,
|
||||
ie8: false,
|
||||
ie: false,
|
||||
inline: true,
|
||||
passes: 2,
|
||||
side_effects: true,
|
||||
@@ -6058,7 +6078,7 @@ issue_3215_4: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
evaluate: true,
|
||||
ie8: true,
|
||||
ie: true,
|
||||
inline: true,
|
||||
passes: 2,
|
||||
side_effects: true,
|
||||
@@ -6950,6 +6970,7 @@ sequence_in_iife_2: {
|
||||
inline: true,
|
||||
passes: 2,
|
||||
side_effects: true,
|
||||
unsafe: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
@@ -6961,8 +6982,7 @@ sequence_in_iife_2: {
|
||||
}
|
||||
expect: {
|
||||
var a = "foo", b = 42;
|
||||
b = a;
|
||||
console.log(a, b);
|
||||
console.log(b = a, b);
|
||||
}
|
||||
expect_stdout: "foo foo"
|
||||
}
|
||||
@@ -6974,6 +6994,7 @@ sequence_in_iife_3: {
|
||||
passes: 2,
|
||||
side_effects: true,
|
||||
toplevel: true,
|
||||
unsafe: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
@@ -7078,6 +7099,7 @@ setter_side_effect: {
|
||||
substitution_assign: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
unsafe: true,
|
||||
}
|
||||
input: {
|
||||
function f1(a, b) {
|
||||
@@ -7098,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);
|
||||
@@ -7903,6 +7924,7 @@ issue_3744: {
|
||||
assign_value_def: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
reduce_vars: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
@@ -7943,6 +7965,7 @@ join_vars_value_def: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
join_vars: true,
|
||||
reduce_vars: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
@@ -7982,6 +8005,7 @@ join_vars_value_def: {
|
||||
var_value_def: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
reduce_vars: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
@@ -8019,6 +8043,7 @@ var_value_def: {
|
||||
mangleable_var: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
reduce_vars: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
@@ -8050,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,
|
||||
@@ -8485,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",
|
||||
@@ -8756,3 +8844,725 @@ issue_4586_2: {
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
issue_4732_1: {
|
||||
options = {
|
||||
booleans: true,
|
||||
collapse_vars: true,
|
||||
evaluate: true,
|
||||
reduce_vars: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var a = 0;
|
||||
(function(b) {
|
||||
var b = a++;
|
||||
var c = b ? b && console.log("PASS") : 0;
|
||||
})(a++);
|
||||
}
|
||||
expect: {
|
||||
var a = 0;
|
||||
(function(b) {
|
||||
(b = a++) && console.log("PASS");
|
||||
})(a++);
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
issue_4732_2: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
conditionals: true,
|
||||
evaluate: true,
|
||||
reduce_vars: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var a = 0;
|
||||
(function(b) {
|
||||
var b = a++;
|
||||
var c = b ? b && console.log("PASS") : 0;
|
||||
})(a++);
|
||||
}
|
||||
expect: {
|
||||
var a = 0;
|
||||
(function(b) {
|
||||
(b = a++) && b && console.log("PASS");
|
||||
})(a++);
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
dot_in_try: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
}
|
||||
input: {
|
||||
var o, a = 6, b = 7, c;
|
||||
try {
|
||||
c = a * b;
|
||||
o.p(c);
|
||||
} catch (e) {
|
||||
console.log(c);
|
||||
}
|
||||
}
|
||||
expect: {
|
||||
var o, a = 6, b = 7, c;
|
||||
try {
|
||||
c = a * b;
|
||||
o.p(c);
|
||||
} catch (e) {
|
||||
console.log(c);
|
||||
}
|
||||
}
|
||||
expect_stdout: "42"
|
||||
}
|
||||
|
||||
dot_non_local: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
}
|
||||
input: {
|
||||
var o, a = 6, b = 7, c;
|
||||
function f() {
|
||||
c = a * b;
|
||||
o.p(c);
|
||||
}
|
||||
try {
|
||||
f();
|
||||
} catch (e) {
|
||||
console.log(c);
|
||||
}
|
||||
}
|
||||
expect: {
|
||||
var o, a = 6, b = 7, c;
|
||||
function f() {
|
||||
c = a * b;
|
||||
o.p(c);
|
||||
}
|
||||
try {
|
||||
f();
|
||||
} catch (e) {
|
||||
console.log(c);
|
||||
}
|
||||
}
|
||||
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;
|
||||
var a = "PASS", b;
|
||||
({
|
||||
get PASS() {
|
||||
a = "FAIL";
|
||||
},
|
||||
})[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;
|
||||
var a = "PASS", b;
|
||||
({
|
||||
get PASS() {
|
||||
a = "FAIL";
|
||||
},
|
||||
})[b = a];
|
||||
log(b);
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
issue_4920_4: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
toplevel: true,
|
||||
}
|
||||
input: {
|
||||
var log = console.log;
|
||||
var o = {
|
||||
get [(a = "FAIL 1", "PASS")]() {
|
||||
a = "FAIL 2";
|
||||
},
|
||||
};
|
||||
var a = "PASS", b;
|
||||
o[b = a];
|
||||
log(b);
|
||||
}
|
||||
expect: {
|
||||
var log = console.log;
|
||||
var o = {
|
||||
get [(a = "FAIL 1", "PASS")]() {
|
||||
a = "FAIL 2";
|
||||
},
|
||||
};
|
||||
var a = "PASS", b;
|
||||
o[b = a];
|
||||
log(b);
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
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"
|
||||
}
|
||||
|
||||
issue_5182: {
|
||||
options = {
|
||||
arrows: true,
|
||||
collapse_vars: true,
|
||||
evaluate: true,
|
||||
hoist_props: true,
|
||||
inline: true,
|
||||
merge_vars: true,
|
||||
passes: 4,
|
||||
reduce_vars: true,
|
||||
sequences: true,
|
||||
side_effects: true,
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var con = console;
|
||||
global.log = con.log;
|
||||
var jump = function(x) {
|
||||
console.log("JUMP:", x * 10);
|
||||
return x + x;
|
||||
};
|
||||
var jump2 = jump;
|
||||
var run = function(x) {
|
||||
console.log("RUN:", x * -10);
|
||||
return x * x;
|
||||
};
|
||||
var run2 = run;
|
||||
var bar = (x, y) => {
|
||||
console.log("BAR:", x + y);
|
||||
return x - y;
|
||||
};
|
||||
var bar2 = bar;
|
||||
var obj = {
|
||||
foo: bar2,
|
||||
go: run2,
|
||||
not_used: jump2,
|
||||
};
|
||||
console.log(obj.foo(1, 2), global.log("PASS"));
|
||||
}
|
||||
expect: {
|
||||
var con = console;
|
||||
global.log = con.log,
|
||||
console.log((console.log("BAR:", 3), -1), global.log("PASS"));
|
||||
}
|
||||
expect_stdout: [
|
||||
"BAR: 3",
|
||||
"PASS",
|
||||
"-1 undefined",
|
||||
]
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
@@ -289,3 +289,18 @@ issue_3689: {
|
||||
}
|
||||
expect_stdout: "00"
|
||||
}
|
||||
|
||||
issue_5145: {
|
||||
options = {
|
||||
strings: true,
|
||||
}
|
||||
input: {
|
||||
var a = [];
|
||||
console.log("" + a + ((a[0] = 4) + "2"));
|
||||
}
|
||||
expect: {
|
||||
var a = [];
|
||||
console.log("" + a + (a[0] = 4) + "2");
|
||||
}
|
||||
expect_stdout: "42"
|
||||
}
|
||||
|
||||
@@ -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,240 @@ 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"
|
||||
}
|
||||
|
||||
issue_5138_1: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
}
|
||||
input: {
|
||||
console.log(function(a, b = a = "FAIL") {
|
||||
return a;
|
||||
}() && "PASS");
|
||||
}
|
||||
expect: {
|
||||
console.log(function(a, b = a = "FAIL") {
|
||||
return a;
|
||||
}() && "PASS");
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
issue_5138_2: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
}
|
||||
input: {
|
||||
console.log(function(a, b = a = "FAIL 1") {
|
||||
return a;
|
||||
}(null, "FAIL 2") || "PASS");
|
||||
}
|
||||
expect: {
|
||||
console.log((null, "PASS"));
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
issue_5192: {
|
||||
options = {
|
||||
dead_code: true,
|
||||
ie: true,
|
||||
}
|
||||
input: {
|
||||
(function a(a, [] = a = "PASS") {
|
||||
console.log(a);
|
||||
})("FAIL");
|
||||
}
|
||||
expect: {
|
||||
(function a(a, [] = a = "PASS") {
|
||||
console.log(a);
|
||||
})("FAIL");
|
||||
}
|
||||
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,
|
||||
@@ -1376,6 +1581,75 @@ hoist_vars: {
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
singleton_1: {
|
||||
options = {
|
||||
pure_getters: true,
|
||||
side_effects: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var [ a ] = "P", b, o = {};
|
||||
[ { 1: o.p } ] = [ "FAIL" ];
|
||||
({ foo: [ o.q ] } = { foo: "S" });
|
||||
[ b = "S" ] = [];
|
||||
console.log(a + o.p + o.q + b);
|
||||
}
|
||||
expect: {
|
||||
var b, a = "P"[0], o = {};
|
||||
o.p = [ "FAIL"["1"] ][0];
|
||||
o.q = { foo: "S"[0] }["foo"];
|
||||
[ b = "S" ] = [];
|
||||
console.log(a + o.p + o.q + b);
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
singleton_2: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
passes: 2,
|
||||
pure_getters: true,
|
||||
side_effects: true,
|
||||
unsafe: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var [ a ] = "P", b, o = {};
|
||||
[ { 1: o.p } ] = [ "FAIL" ];
|
||||
({ foo: [ o.q ] } = { foo: "S" });
|
||||
[ b = "S" ] = [];
|
||||
console.log(a + o.p + o.q + b);
|
||||
}
|
||||
expect: {
|
||||
var b, a = "P", o = {};
|
||||
o.p = "A";
|
||||
o.q = "S";
|
||||
[ b = "S" ] = [];
|
||||
console.log(a + o.p + o.q + b);
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
singleton_side_effects: {
|
||||
options = {
|
||||
side_effects: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
[ 42[console.log("foo")] ] = [ console.log("bar") ];
|
||||
}
|
||||
expect: {
|
||||
[ 42[console.log("foo")] ] = [ console.log("bar") ];
|
||||
}
|
||||
expect_stdout: [
|
||||
"bar",
|
||||
"foo",
|
||||
]
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
issue_4280: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
@@ -1824,7 +2098,7 @@ issue_4321: {
|
||||
|
||||
issue_4323: {
|
||||
options = {
|
||||
ie8: true,
|
||||
ie: true,
|
||||
inline: true,
|
||||
merge_vars: true,
|
||||
reduce_vars: true,
|
||||
@@ -1922,7 +2196,7 @@ issue_4372_2: {
|
||||
}
|
||||
expect: {
|
||||
var a;
|
||||
[ a ] = [ "PASS", "FAIL" ];
|
||||
[ a ] = a = [ "PASS", "FAIL" ];
|
||||
console.log(a);
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
@@ -2556,3 +2830,539 @@ 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"
|
||||
}
|
||||
|
||||
issue_5153_array_assign: {
|
||||
options = {
|
||||
dead_code: true,
|
||||
}
|
||||
input: {
|
||||
var a = function*() {
|
||||
yield b;
|
||||
}(), b;
|
||||
[ b ] = b = a;
|
||||
console.log(a === b ? "PASS" : "FAIL");
|
||||
}
|
||||
expect: {
|
||||
var a = function*() {
|
||||
yield b;
|
||||
}(), b;
|
||||
[ b ] = b = a;
|
||||
console.log(a === b ? "PASS" : "FAIL");
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
issue_5153_array_var: {
|
||||
options = {
|
||||
dead_code: true,
|
||||
}
|
||||
input: {
|
||||
var a = function*() {
|
||||
yield b;
|
||||
}(), [ b ] = b = a;
|
||||
console.log(a === b ? "PASS" : "FAIL");
|
||||
}
|
||||
expect: {
|
||||
var a = function*() {
|
||||
yield b;
|
||||
}(), [ b ] = b = a;
|
||||
console.log(a === b ? "PASS" : "FAIL");
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
issue_5153_object_assign: {
|
||||
options = {
|
||||
dead_code: true,
|
||||
}
|
||||
input: {
|
||||
var a = {
|
||||
get p() {
|
||||
return b;
|
||||
},
|
||||
}, b;
|
||||
({
|
||||
p: b
|
||||
} = b = a);
|
||||
console.log(a === b ? "PASS" : "FAIL");
|
||||
}
|
||||
expect: {
|
||||
var a = {
|
||||
get p() {
|
||||
return b;
|
||||
},
|
||||
}, b;
|
||||
({
|
||||
p: b
|
||||
} = b = a);
|
||||
console.log(a === b ? "PASS" : "FAIL");
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
issue_5153_object_var: {
|
||||
options = {
|
||||
dead_code: true,
|
||||
}
|
||||
input: {
|
||||
var a = {
|
||||
get p() {
|
||||
return b;
|
||||
},
|
||||
}, {
|
||||
p: b
|
||||
} = b = a;
|
||||
console.log(a === b ? "PASS" : "FAIL");
|
||||
}
|
||||
expect: {
|
||||
var a = {
|
||||
get p() {
|
||||
return b;
|
||||
},
|
||||
}, {
|
||||
p: b
|
||||
} = b = a;
|
||||
console.log(a === b ? "PASS" : "FAIL");
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
issue_5168: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
}
|
||||
input: {
|
||||
(function a({
|
||||
[console.log(typeof function() {
|
||||
++a;
|
||||
return a;
|
||||
}())]: b,
|
||||
}) {
|
||||
var a;
|
||||
})({});
|
||||
}
|
||||
expect: {
|
||||
(function a({
|
||||
[console.log(typeof function() {
|
||||
++a;
|
||||
return a;
|
||||
}())]: b,
|
||||
}) {
|
||||
var a;
|
||||
})({});
|
||||
}
|
||||
expect_stdout: "function"
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
issue_5189_1: {
|
||||
options = {
|
||||
pure_getters: "strict",
|
||||
reduce_vars: true,
|
||||
side_effects: true,
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var a = 42;
|
||||
[ a.p ] = a = "PASS";
|
||||
console.log(a);
|
||||
}
|
||||
expect: {
|
||||
var a;
|
||||
[ a.p ] = a = "PASS";
|
||||
console.log(a);
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
issue_5189_2: {
|
||||
options = {
|
||||
pure_getters: "strict",
|
||||
reduce_vars: true,
|
||||
side_effects: true,
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var a = 42;
|
||||
({ p: a.q } = a = "PASS");
|
||||
console.log(a);
|
||||
}
|
||||
expect: {
|
||||
var a;
|
||||
({ p: a.q } = a = "PASS");
|
||||
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"
|
||||
@@ -2916,7 +2916,7 @@ issue_4133: {
|
||||
console.log(a);
|
||||
}
|
||||
expect: {
|
||||
var b = 1;
|
||||
var a = 1;
|
||||
console.log(0);
|
||||
}
|
||||
expect_stdout: "0"
|
||||
@@ -3062,7 +3062,7 @@ issue_4184: {
|
||||
expect_stdout: "42"
|
||||
}
|
||||
|
||||
issue_4235: {
|
||||
issue_4235_1: {
|
||||
options = {
|
||||
inline: true,
|
||||
reduce_vars: true,
|
||||
@@ -3080,12 +3080,35 @@ issue_4235: {
|
||||
})();
|
||||
}
|
||||
expect: {
|
||||
void function() {
|
||||
var f = console.log(f);
|
||||
}();
|
||||
}
|
||||
expect_stdout: "undefined"
|
||||
}
|
||||
|
||||
issue_4235_2: {
|
||||
options = {
|
||||
inline: true,
|
||||
passes: 2,
|
||||
reduce_vars: true,
|
||||
side_effects: true,
|
||||
unused: true,
|
||||
varify: true,
|
||||
}
|
||||
input: {
|
||||
(function() {
|
||||
f = console.log(f),
|
||||
void 0;
|
||||
var f;
|
||||
{
|
||||
const f = 0;
|
||||
}
|
||||
(function f() {
|
||||
var f = console.log(f);
|
||||
})();
|
||||
})();
|
||||
}
|
||||
expect: {
|
||||
console.log(void 0);
|
||||
}
|
||||
expect_stdout: "undefined"
|
||||
}
|
||||
|
||||
@@ -3231,7 +3254,7 @@ issue_4558_1: {
|
||||
issue_4558_2: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
ie8: true,
|
||||
ie: true,
|
||||
reduce_vars: true,
|
||||
unused: true,
|
||||
}
|
||||
@@ -3274,3 +3297,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: {
|
||||
@@ -847,6 +868,8 @@ unsafe_charAt_noop: {
|
||||
unsafe: true,
|
||||
}
|
||||
input: {
|
||||
s = "foo";
|
||||
x = 42;
|
||||
console.log(
|
||||
s.charAt(0),
|
||||
"string".charAt(x),
|
||||
@@ -854,12 +877,15 @@ unsafe_charAt_noop: {
|
||||
);
|
||||
}
|
||||
expect: {
|
||||
s = "foo";
|
||||
x = 42;
|
||||
console.log(
|
||||
s[0],
|
||||
"string"[0 | x],
|
||||
(typeof x)[0]
|
||||
s[0] || "",
|
||||
"string"[0 | x] || "",
|
||||
(typeof x)[0] || ""
|
||||
);
|
||||
}
|
||||
expect_stdout: "f n"
|
||||
}
|
||||
|
||||
issue_1649: {
|
||||
@@ -3176,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,10 +99,49 @@ 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"
|
||||
node_version: ">=8"
|
||||
}
|
||||
|
||||
issue_4715: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
}
|
||||
input: {
|
||||
A = 1;
|
||||
console.log((-0) ** A + 0);
|
||||
console.log((-0) ** A - 0);
|
||||
console.log((-0) ** A * 1);
|
||||
console.log((-0) ** A / 1);
|
||||
console.log(Math.pow(-0, A) + 0);
|
||||
console.log(Math.pow(-0, A) - 0);
|
||||
console.log(Math.pow(-0, A) * 1);
|
||||
console.log(Math.pow(-0, A) / 1);
|
||||
}
|
||||
expect: {
|
||||
A = 1;
|
||||
console.log((-0) ** A + 0);
|
||||
console.log((-0) ** A);
|
||||
console.log((-0) ** A * 1);
|
||||
console.log((-0) ** A);
|
||||
console.log(Math.pow(-0, A) + 0);
|
||||
console.log(+Math.pow(-0, A));
|
||||
console.log(+Math.pow(-0, A));
|
||||
console.log(+Math.pow(-0, A));
|
||||
}
|
||||
expect_stdout: [
|
||||
"0",
|
||||
"-0",
|
||||
"-0",
|
||||
"-0",
|
||||
"0",
|
||||
"-0",
|
||||
"-0",
|
||||
"-0",
|
||||
]
|
||||
node_version: ">=8"
|
||||
}
|
||||
|
||||
@@ -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";
|
||||
@@ -203,7 +210,23 @@ mangle_rename: {
|
||||
}
|
||||
}
|
||||
|
||||
hoist_exports: {
|
||||
hoist_exports_1: {
|
||||
options = {
|
||||
hoist_exports: true,
|
||||
}
|
||||
input: {
|
||||
export { a };
|
||||
export var b;
|
||||
export function f() {}
|
||||
}
|
||||
expect: {
|
||||
var b;
|
||||
function f() {}
|
||||
export { a, b, f };
|
||||
}
|
||||
}
|
||||
|
||||
hoist_exports_2: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
hoist_exports: true,
|
||||
@@ -226,15 +249,15 @@ hoist_exports: {
|
||||
}
|
||||
}
|
||||
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 };
|
||||
}
|
||||
}
|
||||
|
||||
@@ -382,3 +405,94 @@ single_use_class_default: {
|
||||
A.prototype.p = "PASS";
|
||||
}
|
||||
}
|
||||
|
||||
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,
|
||||
}
|
||||
input: {
|
||||
var a = 42;
|
||||
export var a;
|
||||
}
|
||||
expect: {
|
||||
var a = 42;
|
||||
export var a;
|
||||
}
|
||||
}
|
||||
|
||||
issue_4742_join_vars_2: {
|
||||
options = {
|
||||
join_vars: true,
|
||||
}
|
||||
input: {
|
||||
export var a = "foo";
|
||||
var b;
|
||||
b = "bar";
|
||||
}
|
||||
expect: {
|
||||
export var a = "foo";
|
||||
var b, b = "bar";
|
||||
}
|
||||
}
|
||||
|
||||
issue_4742_unused_1: {
|
||||
options = {
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var a = 42;
|
||||
export var a;
|
||||
}
|
||||
expect: {
|
||||
var a = 42;
|
||||
export var a;
|
||||
}
|
||||
}
|
||||
|
||||
issue_4742_unused_2: {
|
||||
options = {
|
||||
reduce_vars: true,
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
export var a = "foo";
|
||||
var a = "bar";
|
||||
}
|
||||
expect: {
|
||||
export var a = "foo";
|
||||
a = "bar";
|
||||
}
|
||||
}
|
||||
|
||||
issue_4761: {
|
||||
input: {
|
||||
export default "function" == 42;
|
||||
}
|
||||
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";
|
||||
}
|
||||
}
|
||||
|
||||
File diff suppressed because it is too large
Load Diff
@@ -217,7 +217,8 @@ name_collision_1: {
|
||||
var obj_foo = 1;
|
||||
var obj_bar = 2;
|
||||
function f() {
|
||||
var obj_foo$0 = 3,
|
||||
var obj,
|
||||
obj_foo$0 = 3,
|
||||
obj_bar = 4,
|
||||
obj_b_r = 5,
|
||||
obj_b_r$0 = 6,
|
||||
@@ -249,7 +250,8 @@ name_collision_2: {
|
||||
console.log(o.p === o.p, o["+"](4), o["-"](5), o__$0, o__$1);
|
||||
}
|
||||
expect: {
|
||||
var o_p = 1,
|
||||
var o,
|
||||
o_p = 1,
|
||||
o__ = function(x) {
|
||||
return x;
|
||||
},
|
||||
@@ -283,7 +285,8 @@ name_collision_3: {
|
||||
console.log(o.p === o.p, o["+"](4), o["-"](5));
|
||||
}
|
||||
expect: {
|
||||
var o_p = 1,
|
||||
var o,
|
||||
o_p = 1,
|
||||
o__ = function(x) {
|
||||
return x;
|
||||
},
|
||||
@@ -315,7 +318,7 @@ name_collision_4: {
|
||||
}
|
||||
expect: {
|
||||
console.log(function() {
|
||||
var o_p$0 = 0, o_q = "PASS";
|
||||
var o, o_p$0 = 0, o_q = "PASS";
|
||||
return function(o_p) {
|
||||
if (!o_p$0) return o_p;
|
||||
}(o_q);
|
||||
@@ -768,7 +771,7 @@ issue_3046: {
|
||||
expect: {
|
||||
console.log(function(a) {
|
||||
do {
|
||||
var b_c = a++;
|
||||
var b, b_c = a++;
|
||||
} while (b_c && a);
|
||||
return a;
|
||||
}(0));
|
||||
@@ -777,6 +780,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,
|
||||
@@ -905,7 +934,7 @@ issue_3411: {
|
||||
expect: {
|
||||
var c = 1;
|
||||
!function f() {
|
||||
var o_p = --c && f();
|
||||
var o, o_p = --c && f();
|
||||
+{} || console.log("PASS");
|
||||
}();
|
||||
}
|
||||
@@ -1016,9 +1045,7 @@ issue_3945_1: {
|
||||
expect: {
|
||||
function f() {
|
||||
o.p;
|
||||
var o = {
|
||||
q: 0,
|
||||
};
|
||||
var o, o_q = 0;
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -1037,9 +1064,7 @@ issue_3945_2: {
|
||||
}
|
||||
expect: {
|
||||
console.log(typeof o);
|
||||
var o = {
|
||||
p: 0,
|
||||
};
|
||||
var o, o_p = 0;
|
||||
}
|
||||
expect_stdout: "undefined"
|
||||
}
|
||||
@@ -1094,3 +1119,60 @@ 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, a_p = 42;
|
||||
console.log(function() {
|
||||
({});
|
||||
}());
|
||||
}
|
||||
expect_stdout: "undefined"
|
||||
}
|
||||
|
||||
issue_5182: {
|
||||
options = {
|
||||
hoist_props: true,
|
||||
merge_vars: true,
|
||||
passes: 2,
|
||||
reduce_vars: true,
|
||||
side_effects: true,
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var o = console;
|
||||
log = o.log;
|
||||
o = {
|
||||
p: function(a) {
|
||||
console.log(a ? "PASS" : "FAIL");
|
||||
return a;
|
||||
},
|
||||
};
|
||||
log(o.p(42));
|
||||
}
|
||||
expect: {
|
||||
var o_p = console;
|
||||
log = o_p.log;
|
||||
o_p = function(a) {
|
||||
console.log(a ? "PASS" : "FAIL");
|
||||
return a;
|
||||
};
|
||||
log(o_p(42));
|
||||
}
|
||||
expect_stdout: [
|
||||
"PASS",
|
||||
"42",
|
||||
]
|
||||
}
|
||||
|
||||
@@ -135,7 +135,7 @@ issue_2295: {
|
||||
}
|
||||
}
|
||||
|
||||
issue_4487: {
|
||||
issue_4487_1: {
|
||||
options = {
|
||||
functions: true,
|
||||
hoist_vars: true,
|
||||
@@ -150,15 +150,64 @@ issue_4487: {
|
||||
};
|
||||
var b = a();
|
||||
}
|
||||
expect: {
|
||||
var a = function f() {
|
||||
var f = console.log(typeof f);
|
||||
};
|
||||
a();
|
||||
}
|
||||
expect_stdout: "undefined"
|
||||
}
|
||||
|
||||
issue_4487_2: {
|
||||
options = {
|
||||
functions: true,
|
||||
hoist_vars: true,
|
||||
keep_fnames: true,
|
||||
passes: 2,
|
||||
reduce_vars: true,
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var a = function f() {
|
||||
var f = console.log(typeof f);
|
||||
};
|
||||
var b = a();
|
||||
}
|
||||
expect: {
|
||||
function a() {
|
||||
var a = console.log(typeof a);
|
||||
var f = console.log(typeof f);
|
||||
}
|
||||
a();
|
||||
}
|
||||
expect_stdout: "undefined"
|
||||
}
|
||||
|
||||
issue_4487_3: {
|
||||
options = {
|
||||
functions: true,
|
||||
hoist_vars: true,
|
||||
keep_fnames: true,
|
||||
passes: 3,
|
||||
reduce_vars: true,
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var a = function f() {
|
||||
var f = console.log(typeof f);
|
||||
};
|
||||
var b = a();
|
||||
}
|
||||
expect: {
|
||||
(function a() {
|
||||
console.log(typeof void 0);
|
||||
})();
|
||||
}
|
||||
expect_stdout: "undefined"
|
||||
}
|
||||
|
||||
issue_4489: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
@@ -207,3 +256,247 @@ issue_4517: {
|
||||
}
|
||||
expect_stdout: "2boolean"
|
||||
}
|
||||
|
||||
issue_4736: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
evaluate: true,
|
||||
hoist_vars: true,
|
||||
merge_vars: true,
|
||||
reduce_vars: true,
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var a;
|
||||
function f() {
|
||||
(function g() {
|
||||
var b = (a = 0, 1 << 30);
|
||||
var c = (a = 0, console.log(b));
|
||||
var d = c;
|
||||
})(f);
|
||||
}
|
||||
f();
|
||||
}
|
||||
expect: {
|
||||
(function() {
|
||||
(function() {
|
||||
0,
|
||||
console.log(1073741824);
|
||||
})();
|
||||
})();
|
||||
}
|
||||
expect_stdout: "1073741824"
|
||||
}
|
||||
|
||||
issue_4839: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
hoist_vars: true,
|
||||
keep_fargs: false,
|
||||
reduce_vars: true,
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var log = console.log, o = function(a, b) {
|
||||
return b && b;
|
||||
}("foo");
|
||||
for (var k in o)
|
||||
throw "FAIL";
|
||||
log("PASS");
|
||||
}
|
||||
expect: {
|
||||
var k, log = console.log;
|
||||
for (k in void 0)
|
||||
throw "FAIL";
|
||||
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) {
|
||||
console.log(Infinity);
|
||||
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"
|
||||
}
|
||||
|
||||
issue_5187: {
|
||||
options = {
|
||||
hoist_props: true,
|
||||
hoist_vars: true,
|
||||
reduce_vars: true,
|
||||
side_effects: true,
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
function f() {
|
||||
var a = 42;
|
||||
do {
|
||||
var b = { 0: a++ };
|
||||
} while (console.log(b[b ^= 0]));
|
||||
}
|
||||
f();
|
||||
}
|
||||
expect: {
|
||||
(function() {
|
||||
var b, a = 42;
|
||||
do {
|
||||
b = { 0: a++ };
|
||||
} while (console.log(b[b ^= 0]));
|
||||
})();
|
||||
}
|
||||
expect_stdout: "42"
|
||||
}
|
||||
|
||||
issue_5195: {
|
||||
options = {
|
||||
hoist_props: true,
|
||||
hoist_vars: true,
|
||||
reduce_vars: true,
|
||||
side_effects: true,
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
function f() {
|
||||
var a;
|
||||
do {
|
||||
var b = { p: a };
|
||||
} while (console.log(b += ""));
|
||||
}
|
||||
f();
|
||||
}
|
||||
expect: {
|
||||
(function() {
|
||||
var a, b;
|
||||
do {
|
||||
b = { p: a };
|
||||
} while (console.log(b += ""));
|
||||
})();
|
||||
}
|
||||
expect_stdout: "[object Object]"
|
||||
}
|
||||
|
||||
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,
|
||||
@@ -463,15 +465,19 @@ drop_fargs: {
|
||||
var a = 1;
|
||||
!function(a_1) {
|
||||
a++;
|
||||
}(a++ + (a && a.var));
|
||||
}(a++ + (a && console.log(a)));
|
||||
console.log(a);
|
||||
}
|
||||
expect: {
|
||||
var a = 1;
|
||||
++a && a.var, a++;
|
||||
++a && console.log(a),
|
||||
a++;
|
||||
console.log(a);
|
||||
}
|
||||
expect_stdout: "3"
|
||||
expect_stdout: [
|
||||
"2",
|
||||
"3",
|
||||
]
|
||||
}
|
||||
|
||||
keep_fargs: {
|
||||
@@ -486,13 +492,17 @@ keep_fargs: {
|
||||
var a = 1;
|
||||
!function(a_1) {
|
||||
a++;
|
||||
}(a++ + (a && a.var));
|
||||
}(a++ + (a && console.log(a)));
|
||||
console.log(a);
|
||||
}
|
||||
expect: {
|
||||
var a = 1;
|
||||
++a && a.var, a++;
|
||||
++a && console.log(a),
|
||||
a++;
|
||||
console.log(a);
|
||||
}
|
||||
expect_stdout: "3"
|
||||
expect_stdout: [
|
||||
"2",
|
||||
"3",
|
||||
]
|
||||
}
|
||||
|
||||
@@ -489,6 +489,116 @@ join_object_assignments_regex: {
|
||||
expect_stdout: "1"
|
||||
}
|
||||
|
||||
chained_assignments: {
|
||||
options = {
|
||||
join_vars: true,
|
||||
}
|
||||
input: {
|
||||
var a, b = a = {};
|
||||
b.p = "PASS";
|
||||
console.log(a.p);
|
||||
}
|
||||
expect: {
|
||||
var a, b = a = {
|
||||
p: "PASS",
|
||||
};
|
||||
console.log(a.p);
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
folded_assignments_1: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
join_vars: true,
|
||||
}
|
||||
input: {
|
||||
var a = {};
|
||||
a[a.PASS = 42] = "PASS";
|
||||
console.log(a[42], a.PASS);
|
||||
}
|
||||
expect: {
|
||||
var a = {
|
||||
PASS: 42,
|
||||
42: "PASS",
|
||||
};
|
||||
console.log(a[42], a.PASS);
|
||||
}
|
||||
expect_stdout: "PASS 42"
|
||||
}
|
||||
|
||||
folded_assignments_2: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
join_vars: true,
|
||||
}
|
||||
input: {
|
||||
"use strict";
|
||||
var a = {};
|
||||
a[42] = "FAIL";
|
||||
a[a.PASS = 42] = "PASS";
|
||||
console.log(a[42], a.PASS);
|
||||
}
|
||||
expect: {
|
||||
"use strict";
|
||||
var a = {
|
||||
42: "FAIL",
|
||||
PASS: 42,
|
||||
};
|
||||
a[42] = "PASS";
|
||||
console.log(a[42], a.PASS);
|
||||
}
|
||||
expect_stdout: "PASS 42"
|
||||
}
|
||||
|
||||
inlined_assignments: {
|
||||
options = {
|
||||
join_vars: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var a;
|
||||
(a = {}).p = "PASS";
|
||||
console.log(a.p);
|
||||
}
|
||||
expect: {
|
||||
var a = {
|
||||
p: "PASS",
|
||||
};
|
||||
console.log(a.p);
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
typescript_enum: {
|
||||
rename = true
|
||||
options = {
|
||||
assignments: true,
|
||||
collapse_vars: true,
|
||||
evaluate: true,
|
||||
hoist_props: true,
|
||||
inline: true,
|
||||
join_vars: true,
|
||||
passes: 4,
|
||||
reduce_vars: true,
|
||||
sequences: true,
|
||||
side_effects: true,
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var Enum;
|
||||
(function (Enum) {
|
||||
Enum[Enum.PASS = 42] = "PASS";
|
||||
})(Enum || (Enum = {}));
|
||||
console.log(Enum[42], Enum.PASS);
|
||||
}
|
||||
expect: {
|
||||
console.log("PASS", 42);
|
||||
}
|
||||
expect_stdout: "PASS 42"
|
||||
}
|
||||
|
||||
issue_2816: {
|
||||
options = {
|
||||
join_vars: true,
|
||||
@@ -1015,7 +1125,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 +1134,7 @@ issue_3856: {
|
||||
expect_stdout: "undefined"
|
||||
}
|
||||
|
||||
issue_3916: {
|
||||
issue_3916_1: {
|
||||
options = {
|
||||
join_vars: true,
|
||||
}
|
||||
@@ -1044,8 +1154,8 @@ issue_3916: {
|
||||
var o = {
|
||||
p: "PASS",
|
||||
__proto__: 42,
|
||||
q: "FAIL",
|
||||
};
|
||||
o.q = "FAIL";
|
||||
o.__proto__ = {
|
||||
p: "FAIL",
|
||||
q: "PASS",
|
||||
@@ -1056,6 +1166,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,
|
||||
@@ -1127,3 +1293,26 @@ assign_sequence_var: {
|
||||
"1 2 3",
|
||||
]
|
||||
}
|
||||
|
||||
issue_5175: {
|
||||
options = {
|
||||
join_vars: true,
|
||||
}
|
||||
input: {
|
||||
function log(f) {
|
||||
console.log(f(), A.p);
|
||||
}
|
||||
log(function() {
|
||||
return (A = {}).p = "PASS";
|
||||
});
|
||||
}
|
||||
expect: {
|
||||
function log(f) {
|
||||
console.log(f(), A.p);
|
||||
}
|
||||
log(function() {
|
||||
return (A = {}).p = "PASS";
|
||||
});
|
||||
}
|
||||
expect_stdout: "PASS PASS"
|
||||
}
|
||||
|
||||
@@ -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);
|
||||
|
||||
@@ -83,8 +83,9 @@ labels_5: {
|
||||
conditionals: true,
|
||||
dead_code: true,
|
||||
if_return: true,
|
||||
unused: true,
|
||||
}
|
||||
// should keep the break-s in the following
|
||||
// should keep `break`s below
|
||||
input: {
|
||||
while (foo) {
|
||||
if (bar) break;
|
||||
@@ -100,8 +101,8 @@ labels_5: {
|
||||
if (bar) break;
|
||||
console.log("foo");
|
||||
}
|
||||
out: while (foo) {
|
||||
if (bar) break out;
|
||||
while (foo) {
|
||||
if (bar) break;
|
||||
console.log("foo");
|
||||
}
|
||||
}
|
||||
@@ -189,23 +190,22 @@ labels_10: {
|
||||
conditionals: true,
|
||||
dead_code: true,
|
||||
if_return: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
out: while (foo) {
|
||||
x();
|
||||
y();
|
||||
out: while (42) {
|
||||
console.log("PASS");
|
||||
break out;
|
||||
z();
|
||||
k();
|
||||
console.log("FAIL");
|
||||
}
|
||||
};
|
||||
expect: {
|
||||
out: while (foo) {
|
||||
x();
|
||||
y();
|
||||
break out;
|
||||
while (42) {
|
||||
console.log("PASS");
|
||||
break;
|
||||
}
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
issue_4466_1: {
|
||||
|
||||
@@ -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,
|
||||
@@ -610,6 +866,28 @@ drop_unused: {
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
default_init: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
reduce_vars: true,
|
||||
sequences: true,
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
"use strict";
|
||||
let a;
|
||||
a = "PASS";
|
||||
console.log(a);
|
||||
}
|
||||
expect: {
|
||||
"use strict";
|
||||
console.log("PASS");
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
issue_4191: {
|
||||
options = {
|
||||
functions: true,
|
||||
@@ -795,6 +1073,7 @@ issue_4210: {
|
||||
issue_4212_1: {
|
||||
options = {
|
||||
dead_code: true,
|
||||
reduce_vars: true,
|
||||
}
|
||||
input: {
|
||||
"use strict";
|
||||
@@ -885,7 +1164,7 @@ issue_4225: {
|
||||
|
||||
issue_4229: {
|
||||
options = {
|
||||
ie8: true,
|
||||
ie: true,
|
||||
side_effects: true,
|
||||
}
|
||||
input: {
|
||||
@@ -916,7 +1195,7 @@ issue_4229: {
|
||||
|
||||
issue_4231: {
|
||||
options = {
|
||||
ie8: true,
|
||||
ie: true,
|
||||
side_effects: true,
|
||||
}
|
||||
input: {
|
||||
@@ -1285,7 +1564,7 @@ issue_4438: {
|
||||
|
||||
issue_4531_1: {
|
||||
mangle = {
|
||||
ie8: true,
|
||||
ie: true,
|
||||
toplevel: true,
|
||||
}
|
||||
input: {
|
||||
@@ -1311,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: {
|
||||
@@ -1404,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,
|
||||
}
|
||||
@@ -476,9 +476,9 @@ issue_4112: {
|
||||
try {
|
||||
throw 42;
|
||||
} catch (e) {
|
||||
var a = e;
|
||||
for (e in a);
|
||||
a = function() {};
|
||||
var o = e;
|
||||
for (e in o);
|
||||
function a() {}
|
||||
console.log(typeof a);
|
||||
return a;
|
||||
}
|
||||
@@ -2620,9 +2620,9 @@ issue_4126_1: {
|
||||
try {
|
||||
console.log("PASS");
|
||||
} catch (e) {
|
||||
var b = a;
|
||||
var c = a;
|
||||
} finally {
|
||||
var c = b;
|
||||
var c = c;
|
||||
}
|
||||
console.log(c);
|
||||
}
|
||||
@@ -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";
|
||||
@@ -3239,3 +3238,199 @@ issue_4653: {
|
||||
"0",
|
||||
]
|
||||
}
|
||||
|
||||
issue_4759: {
|
||||
options = {
|
||||
merge_vars: true,
|
||||
toplevel: true,
|
||||
}
|
||||
input: {
|
||||
var i = 2, a = 1, b, c, d;
|
||||
while (i--) {
|
||||
try {
|
||||
if (1 != b) {
|
||||
d = [];
|
||||
null.p;
|
||||
c = d;
|
||||
} else {
|
||||
b = 0;
|
||||
a = c;
|
||||
}
|
||||
} catch (e) {}
|
||||
b = a;
|
||||
}
|
||||
console.log(a);
|
||||
}
|
||||
expect: {
|
||||
var i = 2, a = 1, b, c, d;
|
||||
while (i--) {
|
||||
try {
|
||||
if (1 != b) {
|
||||
d = [];
|
||||
null.p;
|
||||
c = d;
|
||||
} else {
|
||||
b = 0;
|
||||
a = c;
|
||||
}
|
||||
} catch (e) {}
|
||||
b = a;
|
||||
}
|
||||
console.log(a);
|
||||
}
|
||||
expect_stdout: "undefined"
|
||||
}
|
||||
|
||||
issue_4761: {
|
||||
options = {
|
||||
merge_vars: true,
|
||||
toplevel: true,
|
||||
}
|
||||
input: {
|
||||
var a = "FAIL", b;
|
||||
try {
|
||||
!a && --a && (b = 0)[console] || console.log(b);
|
||||
} catch (e) {}
|
||||
}
|
||||
expect: {
|
||||
var a = "FAIL", b;
|
||||
try {
|
||||
!a && --a && (b = 0)[console] || console.log(b);
|
||||
} catch (e) {}
|
||||
}
|
||||
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"
|
||||
}
|
||||
|
||||
issue_5182: {
|
||||
options = {
|
||||
arrows: true,
|
||||
collapse_vars: true,
|
||||
evaluate: true,
|
||||
hoist_props: true,
|
||||
inline: true,
|
||||
merge_vars: true,
|
||||
passes: 4,
|
||||
reduce_vars: true,
|
||||
sequences: true,
|
||||
side_effects: true,
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
try {
|
||||
var con = console;
|
||||
} catch (x) {}
|
||||
global.log = con.log;
|
||||
var jump = function(x) {
|
||||
console.log("JUMP:", x * 10);
|
||||
return x + x;
|
||||
};
|
||||
var jump2 = jump;
|
||||
var run = function(x) {
|
||||
console.log("RUN:", x * -10);
|
||||
return x * x;
|
||||
};
|
||||
var run2 = run;
|
||||
var bar = (x, y) => {
|
||||
console.log("BAR:", x + y);
|
||||
return x - y;
|
||||
};
|
||||
var bar2 = bar;
|
||||
var obj = {
|
||||
foo: bar2,
|
||||
go: run2,
|
||||
not_used: jump2,
|
||||
};
|
||||
console.log(obj.foo(1, 2), global.log("PASS"));
|
||||
}
|
||||
expect: {
|
||||
try {
|
||||
var con = console;
|
||||
} catch (x) {}
|
||||
global.log = con.log,
|
||||
console.log((console.log("BAR:", 3), -1), global.log("PASS"));
|
||||
}
|
||||
expect_stdout: [
|
||||
"BAR: 3",
|
||||
"PASS",
|
||||
"-1 undefined",
|
||||
]
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
@@ -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,181 @@ 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"
|
||||
}
|
||||
|
||||
object_methods: {
|
||||
options = {
|
||||
properties: true,
|
||||
}
|
||||
input: {
|
||||
({
|
||||
p() {
|
||||
console.log("FAIL 1");
|
||||
},
|
||||
*q() {
|
||||
console.log("FAIL 2");
|
||||
},
|
||||
async r() {
|
||||
console.log("FAIL 3");
|
||||
},
|
||||
async *s() {
|
||||
console.log("PASS");
|
||||
},
|
||||
}).s().next();
|
||||
}
|
||||
expect: {
|
||||
[
|
||||
() => {
|
||||
console.log("FAIL 1");
|
||||
},
|
||||
function*() {
|
||||
console.log("FAIL 2");
|
||||
},
|
||||
async () => {
|
||||
console.log("FAIL 3");
|
||||
},
|
||||
async function*() {
|
||||
console.log("PASS");
|
||||
},
|
||||
][3]().next();
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=10"
|
||||
}
|
||||
|
||||
issue_5177: {
|
||||
options = {
|
||||
properties: true,
|
||||
}
|
||||
input: {
|
||||
var a = "FAIL";
|
||||
var o = { a: "PASS" };
|
||||
o.p = {
|
||||
q() {
|
||||
return this.a;
|
||||
},
|
||||
}.q;
|
||||
console.log(o.p());
|
||||
}
|
||||
expect: {
|
||||
var a = "FAIL";
|
||||
var o = { a: "PASS" };
|
||||
o.p = {
|
||||
q() {
|
||||
return this.a;
|
||||
},
|
||||
}.q;
|
||||
console.log(o.p());
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
@@ -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",
|
||||
]
|
||||
}
|
||||
|
||||
@@ -1220,13 +1220,111 @@ drop_arguments: {
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
lvalues_def: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
pure_getters: true,
|
||||
side_effects: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var a = 0, b = 1;
|
||||
var a = b++, b = +function() {}();
|
||||
a && a[a++];
|
||||
console.log(a, b);
|
||||
}
|
||||
expect: {
|
||||
var a = 0, b = 1;
|
||||
a = b++, b = +void 0;
|
||||
a && a++;
|
||||
console.log(a, b);
|
||||
}
|
||||
expect_stdout: true
|
||||
}
|
||||
|
||||
side_effects_assign: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
pure_getters: true,
|
||||
reduce_funcs: true,
|
||||
reduce_vars: true,
|
||||
sequences: true,
|
||||
side_effects: true,
|
||||
toplevel: true,
|
||||
}
|
||||
input: {
|
||||
var a = typeof void (a && a.in == 1, 0);
|
||||
console.log(a);
|
||||
}
|
||||
expect: {
|
||||
var a = "undefined";
|
||||
console.log(a);
|
||||
}
|
||||
expect_stdout: "undefined"
|
||||
}
|
||||
|
||||
issue_2062: {
|
||||
options = {
|
||||
booleans: true,
|
||||
collapse_vars: true,
|
||||
conditionals: true,
|
||||
pure_getters: true,
|
||||
side_effects: true,
|
||||
}
|
||||
input: {
|
||||
var a = 1;
|
||||
if ([ a || a++ + a--, a++ + a--, a && a.var ]);
|
||||
console.log(a);
|
||||
}
|
||||
expect: {
|
||||
var a = 1;
|
||||
a || (a++, a--), a++, a--;
|
||||
console.log(a);
|
||||
}
|
||||
expect_stdout: "1"
|
||||
}
|
||||
|
||||
issue_2878: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
pure_getters: true,
|
||||
sequences: true,
|
||||
}
|
||||
input: {
|
||||
var c = 0;
|
||||
(function(a, b) {
|
||||
function f2() {
|
||||
if (a) c++;
|
||||
}
|
||||
b = f2();
|
||||
a = 1;
|
||||
b && b.b;
|
||||
f2();
|
||||
})();
|
||||
console.log(c);
|
||||
}
|
||||
expect: {
|
||||
var c = 0;
|
||||
(function(a, b) {
|
||||
function f2() {
|
||||
if (a) c++;
|
||||
}
|
||||
b = f2(),
|
||||
a = 1,
|
||||
f2();
|
||||
})(),
|
||||
console.log(c);
|
||||
}
|
||||
expect_stdout: "1"
|
||||
}
|
||||
|
||||
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,
|
||||
@@ -1242,6 +1340,74 @@ issue_3427: {
|
||||
expect_stdout: true
|
||||
}
|
||||
|
||||
issue_3490_1: {
|
||||
options = {
|
||||
conditionals: true,
|
||||
dead_code: true,
|
||||
inline: true,
|
||||
pure_getters: true,
|
||||
sequences: true,
|
||||
side_effects: true,
|
||||
toplevel: true,
|
||||
}
|
||||
input: {
|
||||
var b = 42, c = "FAIL";
|
||||
if ({
|
||||
3: function() {
|
||||
var a;
|
||||
return (a && a.p) < this;
|
||||
}(),
|
||||
}) c = "PASS";
|
||||
if (b) while ("" == typeof d);
|
||||
console.log(c, b);
|
||||
}
|
||||
expect: {
|
||||
var b = 42, c = "FAIL";
|
||||
if (function() {
|
||||
var a;
|
||||
}(), c = "PASS", b) while ("" == typeof d);
|
||||
console.log(c, b);
|
||||
}
|
||||
expect_stdout: "PASS 42"
|
||||
}
|
||||
|
||||
issue_4135: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
inline: true,
|
||||
merge_vars: true,
|
||||
pure_getters: true,
|
||||
reduce_vars: true,
|
||||
side_effects: true,
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var a = 0, b = 0;
|
||||
--b;
|
||||
a++;
|
||||
if (!a)
|
||||
var c = function() {
|
||||
var d = 0;
|
||||
function f() {
|
||||
d && d.p;
|
||||
}
|
||||
f();
|
||||
this;
|
||||
}(a++);
|
||||
console.log(a, b, c);
|
||||
}
|
||||
expect: {
|
||||
var a = 0;
|
||||
0;
|
||||
a++;
|
||||
if (!a)
|
||||
var c = void a++;
|
||||
console.log(a, -1, c);
|
||||
}
|
||||
expect_stdout: "1 -1 undefined"
|
||||
}
|
||||
|
||||
issue_4440: {
|
||||
options = {
|
||||
pure_getters: "strict",
|
||||
@@ -1270,3 +1436,252 @@ issue_4440: {
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
issue_4730_1: {
|
||||
options = {
|
||||
pure_getters: true,
|
||||
side_effects: true,
|
||||
}
|
||||
input: {
|
||||
var a;
|
||||
console.log("PASS") + (a && a[a.p]);
|
||||
}
|
||||
expect: {
|
||||
var a;
|
||||
console.log("PASS");
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
issue_4730_2: {
|
||||
options = {
|
||||
pure_getters: true,
|
||||
side_effects: true,
|
||||
}
|
||||
input: {
|
||||
var a;
|
||||
!console.log("PASS") || a && a[a.p];
|
||||
}
|
||||
expect: {
|
||||
var a;
|
||||
console.log("PASS");
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
issue_4751: {
|
||||
options = {
|
||||
pure_getters: true,
|
||||
side_effects: true,
|
||||
}
|
||||
input: {
|
||||
var o = {
|
||||
get p() {
|
||||
console.log("PASS");
|
||||
},
|
||||
};
|
||||
o && o.p;
|
||||
}
|
||||
expect: {
|
||||
var o = {
|
||||
get p() {
|
||||
console.log("PASS");
|
||||
},
|
||||
};
|
||||
}
|
||||
}
|
||||
|
||||
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"
|
||||
}
|
||||
|
||||
nested_property_assignments_4: {
|
||||
options = {
|
||||
pure_getters: true,
|
||||
side_effects: true,
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var n, o = { p: { q: { r: "PASS" } } };
|
||||
(n = o.p).r = n.q.r;
|
||||
console.log(o.p.r);
|
||||
}
|
||||
expect: {
|
||||
var n, o = { p: { q: { r: "PASS" } } };
|
||||
(n = o.p).r = n.q.r;
|
||||
console.log(o.p.r);
|
||||
}
|
||||
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);
|
||||
@@ -7384,6 +7320,64 @@ local_assignment_loop: {
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
local_assignment_modified: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
reduce_vars: true,
|
||||
side_effects: true,
|
||||
toplevel: true,
|
||||
}
|
||||
input: {
|
||||
var a;
|
||||
(a = a || {}).p = 42;
|
||||
console.log(a.p);
|
||||
}
|
||||
expect: {
|
||||
var a;
|
||||
(a = {}).p = 42;
|
||||
console.log(a.p);
|
||||
}
|
||||
expect_stdout: "42"
|
||||
}
|
||||
|
||||
local_declaration: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
reduce_vars: true,
|
||||
side_effects: true,
|
||||
toplevel: true,
|
||||
}
|
||||
input: {
|
||||
var a;
|
||||
a || console.log(a = "PASS");
|
||||
}
|
||||
expect: {
|
||||
var a;
|
||||
console.log("PASS");
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
local_definition_modified: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
reduce_vars: true,
|
||||
side_effects: true,
|
||||
toplevel: true,
|
||||
}
|
||||
input: {
|
||||
var a = a || {};
|
||||
a.p = 42;
|
||||
console.log(a.p);
|
||||
}
|
||||
expect: {
|
||||
var a = {};
|
||||
a.p = 42;
|
||||
console.log(a.p);
|
||||
}
|
||||
expect_stdout: "42"
|
||||
}
|
||||
|
||||
issue_3957_1: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
@@ -7499,6 +7493,7 @@ issue_4030: {
|
||||
collapse_vars: true,
|
||||
evaluate: true,
|
||||
reduce_vars: true,
|
||||
side_effects: true,
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
}
|
||||
@@ -7631,3 +7626,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,247 @@ 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",
|
||||
} ][0]);
|
||||
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"] ],
|
||||
} ][0]);
|
||||
}
|
||||
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"
|
||||
}
|
||||
|
||||
issue_5128_1: {
|
||||
options = {
|
||||
inline: true,
|
||||
}
|
||||
input: {
|
||||
console.log(function() {
|
||||
return function f(...[ a ]) {
|
||||
return a;
|
||||
}("PASS");
|
||||
}());
|
||||
}
|
||||
expect: {
|
||||
console.log(function f(...[ a ]) {
|
||||
return a;
|
||||
}("PASS"));
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
issue_5128_2: {
|
||||
options = {
|
||||
inline: true,
|
||||
keep_fnames: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
console.log(function() {
|
||||
return function f(...[ a ]) {
|
||||
return a;
|
||||
}("PASS");
|
||||
}());
|
||||
}
|
||||
expect: {
|
||||
console.log(function f(...[ a ]) {
|
||||
return a;
|
||||
}("PASS"));
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
issue_5165_1: {
|
||||
options = {
|
||||
conditionals: true,
|
||||
dead_code: true,
|
||||
evaluate: true,
|
||||
reduce_vars: true,
|
||||
rests: true,
|
||||
side_effects: true,
|
||||
switches: true,
|
||||
unsafe: true,
|
||||
}
|
||||
input: {
|
||||
console.log(function([ ...a ]) {
|
||||
switch (a) {
|
||||
case a:
|
||||
return "PASS";
|
||||
}
|
||||
}([]));
|
||||
}
|
||||
expect: {
|
||||
console.log(function([ ...a ]) {
|
||||
return "PASS";
|
||||
}([]));
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
issue_5165_2: {
|
||||
options = {
|
||||
conditionals: true,
|
||||
dead_code: true,
|
||||
evaluate: true,
|
||||
reduce_vars: true,
|
||||
rests: true,
|
||||
side_effects: true,
|
||||
switches: true,
|
||||
unsafe: true,
|
||||
}
|
||||
input: {
|
||||
console.log(function(...a) {
|
||||
switch (a) {
|
||||
case a:
|
||||
return "PASS";
|
||||
}
|
||||
}());
|
||||
}
|
||||
expect: {
|
||||
console.log("PASS");
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
@@ -172,3 +172,49 @@ 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"
|
||||
}
|
||||
|
||||
issue_5197: {
|
||||
rename = true
|
||||
input: {
|
||||
function f(async) {
|
||||
async(")=>{}");
|
||||
}
|
||||
console.log("" + this.__proto__);
|
||||
}
|
||||
expect: {
|
||||
function f(a) {
|
||||
a(")=>{}");
|
||||
}
|
||||
console.log("" + this.__proto__);
|
||||
}
|
||||
expect_stdout: "[object global]"
|
||||
}
|
||||
|
||||
@@ -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,
|
||||
@@ -663,12 +720,21 @@ side_effects_cascade_1: {
|
||||
if (a < 0) a = 0;
|
||||
b.a = a;
|
||||
}
|
||||
var m = {}, n = {};
|
||||
f(13, m);
|
||||
f("foo", n);
|
||||
console.log(m.a, n.a);
|
||||
}
|
||||
expect: {
|
||||
function f(a, b) {
|
||||
(a -= 42) < 0 && (a = 0), b.a = a;
|
||||
b.a = a = (a -= 42) < 0 ? 0 : a;
|
||||
}
|
||||
var m = {}, n = {};
|
||||
f(13, m),
|
||||
f("foo", n),
|
||||
console.log(m.a, n.a);
|
||||
}
|
||||
expect_stdout: "0 NaN"
|
||||
}
|
||||
|
||||
side_effects_cascade_2: {
|
||||
|
||||
@@ -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,
|
||||
@@ -558,3 +588,60 @@ drop_side_effect_free_call: {
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
issue_4730_1: {
|
||||
options = {
|
||||
pure_getters: "strict",
|
||||
side_effects: true,
|
||||
}
|
||||
input: {
|
||||
var a;
|
||||
console.log("PASS") + (a && a[a.p]);
|
||||
}
|
||||
expect: {
|
||||
var a;
|
||||
console.log("PASS"),
|
||||
a && a[a.p];
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
issue_4730_2: {
|
||||
options = {
|
||||
pure_getters: "strict",
|
||||
side_effects: true,
|
||||
}
|
||||
input: {
|
||||
var a;
|
||||
!console.log("PASS") || a && a[a.p];
|
||||
}
|
||||
expect: {
|
||||
var a;
|
||||
!console.log("PASS") || a && a[a.p];
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
issue_4751: {
|
||||
options = {
|
||||
pure_getters: "strict",
|
||||
side_effects: true,
|
||||
}
|
||||
input: {
|
||||
var o = {
|
||||
get p() {
|
||||
console.log("PASS");
|
||||
},
|
||||
};
|
||||
o && o.p;
|
||||
}
|
||||
expect: {
|
||||
var o = {
|
||||
get p() {
|
||||
console.log("PASS");
|
||||
},
|
||||
};
|
||||
o && o.p;
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
}
|
||||
|
||||
@@ -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) {
|
||||
@@ -85,6 +131,105 @@ malformed_escape: {
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
booleans: {
|
||||
options = {
|
||||
booleans: true,
|
||||
evaluate: true,
|
||||
templates: true,
|
||||
}
|
||||
input: {
|
||||
var a;
|
||||
console.log(`$${a}${a}` ? "PASS" : "FAIL");
|
||||
}
|
||||
expect: {
|
||||
var a;
|
||||
console.log("$" + a + a ? "PASS" : "FAIL");
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
escape_placeholder_1: {
|
||||
options = {
|
||||
templates: true,
|
||||
}
|
||||
input: {
|
||||
console.log(`\${\n`);
|
||||
}
|
||||
expect: {
|
||||
console.log(`\${
|
||||
`);
|
||||
}
|
||||
expect_stdout: [
|
||||
"${",
|
||||
"",
|
||||
]
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
escape_placeholder_2: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
templates: true,
|
||||
}
|
||||
input: {
|
||||
console.log(`\n${"${"}\n`);
|
||||
}
|
||||
expect: {
|
||||
console.log(`
|
||||
\${
|
||||
`);
|
||||
}
|
||||
expect_stdout: [
|
||||
"",
|
||||
"${",
|
||||
"",
|
||||
]
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
escape_placeholder_3: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
templates: true,
|
||||
}
|
||||
input: {
|
||||
console.log(`\n$${"{"}\n`);
|
||||
}
|
||||
expect: {
|
||||
console.log(`
|
||||
\${
|
||||
`);
|
||||
}
|
||||
expect_stdout: [
|
||||
"",
|
||||
"${",
|
||||
"",
|
||||
]
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
escape_placeholder_4: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
templates: true,
|
||||
}
|
||||
input: {
|
||||
console.log(`\n${"$"}${"{"}\n`);
|
||||
}
|
||||
expect: {
|
||||
console.log(`
|
||||
\${
|
||||
`);
|
||||
}
|
||||
expect_stdout: [
|
||||
"",
|
||||
"${",
|
||||
"",
|
||||
]
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
evaluate: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
@@ -128,7 +273,7 @@ partial_evaluate: {
|
||||
console.log(`${6 * 7} foo ${console ? `PA` + "SS" : `FA` + `IL`}`);
|
||||
}
|
||||
expect: {
|
||||
console.log(`42 foo ${console ? "PASS" : "FAIL"}`);
|
||||
console.log("42 foo " + (console ? "PASS" : "FAIL"));
|
||||
}
|
||||
expect_stdout: "42 foo PASS"
|
||||
node_version: ">=4"
|
||||
@@ -158,7 +303,7 @@ malformed_evaluate_2: {
|
||||
console.log(`\u0${0}b${5}`);
|
||||
}
|
||||
expect: {
|
||||
console.log(`\u0${0}b5`);
|
||||
console.log(`\u00b` + 5);
|
||||
}
|
||||
expect_stdout: true
|
||||
node_version: ">=4"
|
||||
@@ -211,7 +356,7 @@ unsafe_evaluate: {
|
||||
node_version: ">=8"
|
||||
}
|
||||
|
||||
side_effects: {
|
||||
side_effects_1: {
|
||||
options = {
|
||||
side_effects: true,
|
||||
}
|
||||
@@ -228,6 +373,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 +414,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,
|
||||
@@ -272,13 +456,14 @@ issue_4604: {
|
||||
issue_4606: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
strings: true,
|
||||
templates: true,
|
||||
}
|
||||
input: {
|
||||
console.log(`${typeof A} ${"\r"} ${"\\"} ${"`"}`);
|
||||
}
|
||||
expect: {
|
||||
console.log(`${typeof A} \r \\ \``);
|
||||
console.log(typeof A + " \r \\ `");
|
||||
}
|
||||
expect_stdout: "undefined \r \\ `"
|
||||
node_version: ">=4"
|
||||
@@ -305,7 +490,7 @@ issue_4676: {
|
||||
reduce_vars: true,
|
||||
templates: true,
|
||||
toplevel: true,
|
||||
unsafe:true,
|
||||
unsafe: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
@@ -328,3 +513,259 @@ 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"
|
||||
}
|
||||
|
||||
issue_5125_1: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
strings: true,
|
||||
templates: true,
|
||||
}
|
||||
input: {
|
||||
console.log(`PASS ${typeof A}`);
|
||||
}
|
||||
expect: {
|
||||
console.log("PASS " + typeof A);
|
||||
}
|
||||
expect_stdout: "PASS undefined"
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
issue_5125_2: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
strings: true,
|
||||
templates: true,
|
||||
}
|
||||
input: {
|
||||
console.log(`PASS
|
||||
${typeof A}`);
|
||||
}
|
||||
expect: {
|
||||
console.log(`PASS
|
||||
` + typeof A);
|
||||
}
|
||||
expect_stdout: [
|
||||
"PASS",
|
||||
"undefined",
|
||||
]
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
issue_5125_3: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
strings: true,
|
||||
templates: true,
|
||||
}
|
||||
input: {
|
||||
console.log(`PASS\n${typeof A}`);
|
||||
}
|
||||
expect: {
|
||||
console.log(`PASS
|
||||
` + typeof A);
|
||||
}
|
||||
expect_stdout: [
|
||||
"PASS",
|
||||
"undefined",
|
||||
]
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
issue_5125_4: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
strings: true,
|
||||
templates: true,
|
||||
}
|
||||
input: {
|
||||
console.log(`PASS
|
||||
|
||||
${typeof A}`);
|
||||
}
|
||||
expect: {
|
||||
console.log(`PASS
|
||||
|
||||
` + typeof A);
|
||||
}
|
||||
expect_stdout: [
|
||||
"PASS",
|
||||
"",
|
||||
"undefined",
|
||||
]
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
issue_5125_5: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
strings: true,
|
||||
templates: true,
|
||||
}
|
||||
input: {
|
||||
console.log(`PASS\n\n${typeof A}`);
|
||||
}
|
||||
expect: {
|
||||
console.log(`PASS
|
||||
|
||||
` + typeof A);
|
||||
}
|
||||
expect_stdout: [
|
||||
"PASS",
|
||||
"",
|
||||
"undefined",
|
||||
]
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
issue_5125_6: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
strings: true,
|
||||
templates: true,
|
||||
}
|
||||
input: {
|
||||
console.log(`${typeof A} ${typeof B} PASS`);
|
||||
}
|
||||
expect: {
|
||||
console.log(typeof A + ` ${typeof B} PASS`);
|
||||
}
|
||||
expect_stdout: "undefined undefined PASS"
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
issue_5125_7: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
strings: true,
|
||||
templates: true,
|
||||
}
|
||||
input: {
|
||||
console.log(`${typeof A} ${typeof B} ${typeof C} PASS`);
|
||||
}
|
||||
expect: {
|
||||
console.log(typeof A + ` ${typeof B} ${typeof C} PASS`);
|
||||
}
|
||||
expect_stdout: "undefined undefined undefined PASS"
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
issue_5125_8: {
|
||||
options = {
|
||||
evaluate: true,
|
||||
strings: true,
|
||||
templates: true,
|
||||
}
|
||||
input: {
|
||||
console.log(`${typeof A}${typeof B}${typeof C} PASS`);
|
||||
}
|
||||
expect: {
|
||||
console.log(typeof A + typeof B + typeof C + " PASS");
|
||||
}
|
||||
expect_stdout: "undefinedundefinedundefined PASS"
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
issue_5136: {
|
||||
options = {
|
||||
templates: true,
|
||||
}
|
||||
input: {
|
||||
console.log(`${A = []}${A[0] = 42}`);
|
||||
}
|
||||
expect: {
|
||||
console.log(`` + (A = []) + (A[0] = 42));
|
||||
}
|
||||
expect_stdout: "42"
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
issue_5145_1: {
|
||||
options = {
|
||||
strings: true,
|
||||
templates: true,
|
||||
}
|
||||
input: {
|
||||
var a = [];
|
||||
console.log(`${a}${a[0] = 42}
|
||||
`);
|
||||
}
|
||||
expect: {
|
||||
var a = [];
|
||||
console.log(`${a}${a[0] = 42}
|
||||
`);
|
||||
}
|
||||
expect_stdout: [
|
||||
"42",
|
||||
"",
|
||||
]
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
issue_5145_2: {
|
||||
options = {
|
||||
strings: true,
|
||||
templates: true,
|
||||
}
|
||||
input: {
|
||||
var a = [];
|
||||
console.log(`${a}${a}${a[0] = 42}
|
||||
`);
|
||||
}
|
||||
expect: {
|
||||
var a = [];
|
||||
console.log("" + a + a + (a[0] = 42) + `
|
||||
`);
|
||||
}
|
||||
expect_stdout: [
|
||||
"42",
|
||||
"",
|
||||
]
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
issue_5199: {
|
||||
options = {
|
||||
collapse_vars: true,
|
||||
toplevel: true,
|
||||
unused: true,
|
||||
}
|
||||
input: {
|
||||
var a = function() {
|
||||
console.log(typeof b);
|
||||
}``;
|
||||
{
|
||||
const b = a;
|
||||
}
|
||||
}
|
||||
expect: {
|
||||
var a = function() {
|
||||
console.log(typeof b);
|
||||
}``;
|
||||
{
|
||||
const b = a;
|
||||
}
|
||||
}
|
||||
expect_stdout: "undefined"
|
||||
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,
|
||||
|
||||
@@ -111,7 +111,7 @@ hoist_props_const: {
|
||||
}
|
||||
}
|
||||
expect: {
|
||||
var o_p = "PASS";
|
||||
var o = 0, o_p = "PASS";
|
||||
console.log(o_p);
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
@@ -136,7 +136,7 @@ hoist_props_let: {
|
||||
}
|
||||
expect: {
|
||||
"use strict";
|
||||
var o_p = "PASS";
|
||||
var o, o_p = "PASS";
|
||||
console.log(o_p);
|
||||
}
|
||||
expect_stdout: "PASS"
|
||||
@@ -354,6 +354,92 @@ forin_let_2: {
|
||||
node_version: ">=6"
|
||||
}
|
||||
|
||||
loop_scope_1: {
|
||||
options = {
|
||||
toplevel: true,
|
||||
varify: true,
|
||||
}
|
||||
input: {
|
||||
"use strict";
|
||||
var o = { foo: 1, bar: 2 };
|
||||
for (let i in o) {
|
||||
console.log(i);
|
||||
}
|
||||
for (const j in o)
|
||||
setTimeout(() => console.log(j), 0);
|
||||
for (let k in o)
|
||||
setTimeout(function() {
|
||||
console.log(k);
|
||||
}, 0);
|
||||
}
|
||||
expect: {
|
||||
"use strict";
|
||||
var o = { foo: 1, bar: 2 };
|
||||
for (var i in o)
|
||||
console.log(i);
|
||||
for (const j in o)
|
||||
setTimeout(() => console.log(j), 0);
|
||||
for (let k in o)
|
||||
setTimeout(function() {
|
||||
console.log(k);
|
||||
}, 0);
|
||||
}
|
||||
expect_stdout: [
|
||||
"foo",
|
||||
"bar",
|
||||
"foo",
|
||||
"bar",
|
||||
"foo",
|
||||
"bar",
|
||||
]
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
loop_scope_2: {
|
||||
options = {
|
||||
reduce_vars: true,
|
||||
toplevel: true,
|
||||
varify: true,
|
||||
}
|
||||
input: {
|
||||
"use strict";
|
||||
var a = [ "foo", "bar" ];
|
||||
for (var i = 0; i < a.length; i++) {
|
||||
const x = a[i];
|
||||
console.log(x);
|
||||
let y = a[i];
|
||||
setTimeout(() => console.log(y), 0);
|
||||
const z = a[i];
|
||||
setTimeout(function() {
|
||||
console.log(z);
|
||||
}, 0);
|
||||
}
|
||||
}
|
||||
expect: {
|
||||
"use strict";
|
||||
var a = [ "foo", "bar" ];
|
||||
for (var i = 0; i < a.length; i++) {
|
||||
var x = a[i];
|
||||
console.log(x);
|
||||
let y = a[i];
|
||||
setTimeout(() => console.log(y), 0);
|
||||
const z = a[i];
|
||||
setTimeout(function() {
|
||||
console.log(z);
|
||||
}, 0);
|
||||
}
|
||||
}
|
||||
expect_stdout: [
|
||||
"foo",
|
||||
"bar",
|
||||
"foo",
|
||||
"foo",
|
||||
"bar",
|
||||
"bar",
|
||||
]
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
issue_4290_1_const: {
|
||||
options = {
|
||||
reduce_vars: true,
|
||||
@@ -409,3 +495,121 @@ drop_forin_let: {
|
||||
expect_stdout: "PASS"
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
default_init: {
|
||||
options = {
|
||||
join_vars: true,
|
||||
reduce_vars: true,
|
||||
unused: true,
|
||||
varify: true,
|
||||
}
|
||||
input: {
|
||||
A = "PASS";
|
||||
(function() {
|
||||
"use strict";
|
||||
let a;
|
||||
a = A;
|
||||
console.log(a);
|
||||
})();
|
||||
}
|
||||
expect: {
|
||||
A = "PASS";
|
||||
(function() {
|
||||
"use strict";
|
||||
var a = A;
|
||||
console.log(a);
|
||||
})();
|
||||
}
|
||||
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: {
|
||||
|
||||
@@ -96,7 +96,7 @@ pause_resume: {
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
arrow_yield: {
|
||||
arrow_yield_1: {
|
||||
input: {
|
||||
yield = "PASS";
|
||||
console.log(function*() {
|
||||
@@ -108,6 +108,18 @@ arrow_yield: {
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
arrow_yield_2: {
|
||||
input: {
|
||||
console.log(typeof function *() {
|
||||
// Syntax error on Node.js v6+
|
||||
return (yield) => {};
|
||||
}().next().value);
|
||||
}
|
||||
expect_exact: "console.log(typeof function*(){return(yield)=>{}}().next().value);"
|
||||
expect_stdout: "function"
|
||||
node_version: "4"
|
||||
}
|
||||
|
||||
for_of: {
|
||||
input: {
|
||||
function* f() {
|
||||
@@ -150,6 +162,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 +280,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 +445,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 +503,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 +522,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 +1055,257 @@ 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"
|
||||
}
|
||||
|
||||
issue_5177: {
|
||||
options = {
|
||||
properties: true,
|
||||
}
|
||||
input: {
|
||||
console.log(typeof function*() {
|
||||
return {
|
||||
p(yield) {},
|
||||
}.p;
|
||||
}().next().value);
|
||||
}
|
||||
expect: {
|
||||
console.log(typeof function*() {
|
||||
return {
|
||||
p(yield) {},
|
||||
}.p;
|
||||
}().next().value);
|
||||
}
|
||||
expect_stdout: "function"
|
||||
node_version: ">=4"
|
||||
}
|
||||
|
||||
@@ -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": "/^_/"
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@@ -1,7 +1,5 @@
|
||||
// (beautified)
|
||||
console.log(function() {
|
||||
return 1 + .1 + .1;
|
||||
}());
|
||||
console.log(1 + .1 + .1);
|
||||
// output: 1.2000000000000002
|
||||
//
|
||||
// minify: 1.2
|
||||
|
||||
@@ -58,22 +58,31 @@ 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",
|
||||
"is-my-json-valid@2.20.5",
|
||||
"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,
|
||||
|
||||
@@ -191,7 +191,7 @@ describe("comments", function() {
|
||||
});
|
||||
|
||||
it("Should correctly preserve new lines around comments", function() {
|
||||
var tests = [
|
||||
[
|
||||
[
|
||||
"// foo",
|
||||
"// bar",
|
||||
@@ -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,7 +1,6 @@
|
||||
var assert = require("assert");
|
||||
var exec = require("child_process").exec;
|
||||
var path = require("path");
|
||||
var readFileSync = require("fs").readFileSync;
|
||||
|
||||
describe("bin/uglifyjs with input file globs", function() {
|
||||
var uglifyjscmd = '"' + process.argv[0] + '" bin/uglifyjs';
|
||||
|
||||
@@ -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}'
|
||||
);
|
||||
});
|
||||
});
|
||||
@@ -1,5 +1,5 @@
|
||||
var assert = require("assert");
|
||||
var UglifyJS = require("../..");
|
||||
var UglifyJS = require("../node");
|
||||
|
||||
describe("let", function() {
|
||||
this.timeout(30000);
|
||||
@@ -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,
|
||||
}
|
||||
@@ -54,4 +54,43 @@ describe("let", function() {
|
||||
assert.notStrictEqual(result.indexOf('v["' + name + '"]'), -1);
|
||||
});
|
||||
});
|
||||
it("Should parse `let` as name correctly", function() {
|
||||
[
|
||||
"for(var let;let;let)let;",
|
||||
"function let(let){let}",
|
||||
].forEach(function(code) {
|
||||
var ast = UglifyJS.parse(code);
|
||||
assert.strictEqual(ast.print_to_string(), code);
|
||||
assert.throws(function() {
|
||||
UglifyJS.parse('"use strict";' + code);
|
||||
}, function(e) {
|
||||
return e instanceof UglifyJS.JS_Parse_Error && e.message == "Unexpected let in strict mode";
|
||||
}, code);
|
||||
});
|
||||
});
|
||||
it("Should throw on ambiguous use of `let`", function() {
|
||||
[
|
||||
"export let",
|
||||
[
|
||||
"let",
|
||||
"console.log(42)",
|
||||
].join("\n"),
|
||||
[
|
||||
"let",
|
||||
"[ console.log(42) ]",
|
||||
].join("\n"),
|
||||
[
|
||||
"let",
|
||||
"{",
|
||||
" console.log(42)",
|
||||
"}",
|
||||
].join("\n"),
|
||||
].forEach(function(code) {
|
||||
assert.throws(function() {
|
||||
UglifyJS.parse(code);
|
||||
}, function(e) {
|
||||
return e instanceof UglifyJS.JS_Parse_Error;
|
||||
}, code);
|
||||
});
|
||||
});
|
||||
});
|
||||
|
||||
@@ -25,7 +25,7 @@ describe("minify", function() {
|
||||
if (result.error) throw result.error;
|
||||
assert.strictEqual(result.code, "print(42);");
|
||||
assert.strictEqual(JSON.stringify(options), value);
|
||||
})
|
||||
});
|
||||
it("Should skip inherited keys from `files`", function() {
|
||||
var files = Object.create({ skip: this });
|
||||
files[0] = "alert(1 + 1)";
|
||||
@@ -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() {
|
||||
|
||||
@@ -1,5 +1,4 @@
|
||||
var assert = require("assert");
|
||||
var exec = require("child_process").exec;
|
||||
var fs = require("fs");
|
||||
var reduce_test = require("../reduce");
|
||||
var semver = require("semver");
|
||||
@@ -183,6 +182,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 +281,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 +383,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 = {};",
|
||||
@@ -349,7 +362,7 @@ describe("sourcemaps", function() {
|
||||
it("Should not modify input source map", function() {
|
||||
var orig = get_map();
|
||||
var original = JSON.stringify(orig);
|
||||
var map = prepare_map(orig);
|
||||
prepare_map(orig);
|
||||
assert.strictEqual(JSON.stringify(orig), original);
|
||||
});
|
||||
it("Should copy over original sourcesContent", function() {
|
||||
|
||||
@@ -4,34 +4,27 @@ var UglifyJS = require("../..");
|
||||
|
||||
describe("spidermonkey export/import sanity test", function() {
|
||||
it("Should produce a functional build when using --self with spidermonkey", function(done) {
|
||||
this.timeout(60000);
|
||||
|
||||
this.timeout(120000);
|
||||
var uglifyjs = '"' + process.argv[0] + '" bin/uglifyjs';
|
||||
var command = uglifyjs + " --self -cm --wrap SpiderUglify -o spidermonkey | " +
|
||||
uglifyjs + " -p spidermonkey -cm";
|
||||
|
||||
exec(command, {
|
||||
maxBuffer: 1048576
|
||||
}, function(err, stdout) {
|
||||
var command = [
|
||||
uglifyjs + " --self -cm --wrap SpiderUglify -o spidermonkey",
|
||||
uglifyjs + " -p spidermonkey -cm",
|
||||
].join(" | ");
|
||||
exec(command, { maxBuffer: 1048576 }, function(err, stdout) {
|
||||
if (err) throw err;
|
||||
|
||||
eval(stdout);
|
||||
assert.strictEqual(typeof SpiderUglify, "object");
|
||||
var result = SpiderUglify.minify("foo([true,,2+3]);");
|
||||
assert.strictEqual(result.error, undefined);
|
||||
assert.strictEqual(result.code, "foo([!0,,5]);");
|
||||
|
||||
done();
|
||||
});
|
||||
});
|
||||
|
||||
it("Should not add unnecessary escape slashes to regexps", function() {
|
||||
it("Should not add unnecessary escape slashes to RegExp", function() {
|
||||
var input = "/[\\\\/]/;";
|
||||
var ast = UglifyJS.parse(input).to_mozilla_ast();
|
||||
assert.equal(
|
||||
UglifyJS.AST_Node.from_mozilla_ast(ast).print_to_string(),
|
||||
input
|
||||
);
|
||||
assert.strictEqual(UglifyJS.AST_Node.from_mozilla_ast(ast).print_to_string(), input);
|
||||
});
|
||||
|
||||
it("Should judge between directives and strings correctly on import", function() {
|
||||
|
||||
@@ -12,7 +12,7 @@ describe("String literals", function() {
|
||||
'"\u2029"',
|
||||
].forEach(function(input) {
|
||||
assert.throws(function() {
|
||||
var ast = UglifyJS.parse(input);
|
||||
UglifyJS.parse(input);
|
||||
}, function(e) {
|
||||
return e instanceof UglifyJS.JS_Parse_Error
|
||||
&& e.message === "Unterminated string constant";
|
||||
@@ -44,7 +44,7 @@ describe("String literals", function() {
|
||||
'"use strict";\n"\\011"',
|
||||
].forEach(function(input) {
|
||||
assert.throws(function() {
|
||||
var output = UglifyJS.parse(input);
|
||||
UglifyJS.parse(input);
|
||||
}, function(e) {
|
||||
return e instanceof UglifyJS.JS_Parse_Error
|
||||
&& e.message === "Legacy octal escape sequences are not allowed in strict mode";
|
||||
|
||||
@@ -34,20 +34,20 @@ describe("Template literals", function() {
|
||||
[
|
||||
// native line breaks
|
||||
[ "`foo\nbar`", "`foo\nbar`" ],
|
||||
[ "`foo\rbar`", "`foo\rbar`" ],
|
||||
[ "`foo\rbar`", "`foo\nbar`" ],
|
||||
[ "`foo\r\nbar`", "`foo\nbar`" ],
|
||||
[ "`foo\r\n\rbar`", "`foo\n\rbar`" ],
|
||||
[ "`foo\r\n\rbar`", "`foo\n\nbar`" ],
|
||||
// escaped line breaks
|
||||
[ "`foo\\nbar`", "`foo\\nbar`" ],
|
||||
[ "`foo\\rbar`", "`foo\\rbar`" ],
|
||||
[ "`foo\r\\nbar`", "`foo\r\\nbar`" ],
|
||||
[ "`foo\r\\nbar`", "`foo\n\\nbar`" ],
|
||||
[ "`foo\\r\nbar`", "`foo\\r\nbar`" ],
|
||||
[ "`foo\\r\\nbar`", "`foo\\r\\nbar`" ],
|
||||
// continuation
|
||||
[ "`foo\\\nbar`", "`foo\\\nbar`" ],
|
||||
[ "`foo\\\rbar`", "`foo\\\rbar`" ],
|
||||
[ "`foo\\\rbar`", "`foo\\\nbar`" ],
|
||||
[ "`foo\\\r\nbar`", "`foo\\\nbar`" ],
|
||||
[ "`foo\\\r\n\rbar`", "`foo\\\n\rbar`" ],
|
||||
[ "`foo\\\r\n\rbar`", "`foo\\\n\nbar`" ],
|
||||
[ "`foo\\\\nbar`", "`foo\\\\nbar`" ],
|
||||
[ "`foo\\\\rbar`", "`foo\\\\rbar`" ],
|
||||
[ "`foo\\\\r\nbar`", "`foo\\\\r\nbar`" ],
|
||||
|
||||
@@ -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,15 +132,26 @@ 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;
|
||||
if (parent instanceof U.AST_ExportForeign) return;
|
||||
if (parent instanceof U.AST_ExportReferences) return;
|
||||
// preserve sole definition of an export statement
|
||||
if (node instanceof U.AST_VarDef
|
||||
&& parent.definitions.length == 1
|
||||
&& tt.parent(1) instanceof U.AST_ExportDeclaration) {
|
||||
return;
|
||||
}
|
||||
// preserve for (var xxx; ...)
|
||||
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
|
||||
|
||||
@@ -202,13 +213,11 @@ 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) {
|
||||
var expr = expr instanceof U.AST_Exit ? node.value : node.body;
|
||||
if (expr instanceof U.AST_Node && !U.is_statement(expr)) {
|
||||
// collect expressions from each statements' body
|
||||
node.expression.body.forEach(function(node) {
|
||||
var expr = node instanceof U.AST_Exit ? node.value : node.body;
|
||||
if (expr instanceof U.AST_Node && !U.is_statement(expr) && can_hoist(expr)) {
|
||||
// collect expressions from each statement's body
|
||||
seq.push(expr);
|
||||
}
|
||||
});
|
||||
@@ -258,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 = [];
|
||||
@@ -312,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;
|
||||
@@ -375,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);
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -387,7 +395,7 @@ module.exports = function reduce_test(testcase, minify_options, reduce_options)
|
||||
var expr = [
|
||||
node.expression, // switch expression
|
||||
node.body[0] && node.body[0].expression, // first case expression or undefined
|
||||
node.body[0] && node.body[0], // first case body or undefined
|
||||
node.body[0], // first case body or undefined
|
||||
][ (node.start._permute * steps | 0) % 4 ];
|
||||
node.start._permute += step;
|
||||
if (expr && (!(expr instanceof U.AST_Statement) || !has_loopcontrol(expr, node, parent))) {
|
||||
@@ -460,13 +468,6 @@ module.exports = function reduce_test(testcase, minify_options, reduce_options)
|
||||
return List.skip;
|
||||
}
|
||||
|
||||
// preserve sole definition of an export statement
|
||||
if (node instanceof U.AST_VarDef
|
||||
&& parent.definitions.length == 1
|
||||
&& tt.parent(1) instanceof U.AST_ExportDeclaration) {
|
||||
return node;
|
||||
}
|
||||
|
||||
// remove this node unless its the sole element of a (transient) sequence
|
||||
if (!(parent instanceof U.AST_Sequence) || parent.expressions.length > 1) {
|
||||
node.start._permute++;
|
||||
@@ -489,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;
|
||||
@@ -674,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);
|
||||
}
|
||||
@@ -707,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) {
|
||||
|
||||
105
test/release/acorn.sh
Executable file
105
test/release/acorn.sh
Executable file
@@ -0,0 +1,105 @@
|
||||
#!/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/acorn \
|
||||
&& git clone https://github.com/acornjs/acorn.git tmp/acorn \
|
||||
&& cd tmp/acorn \
|
||||
&& rm -rf .git/hooks \
|
||||
&& git checkout 74b59384320ced82e09da2e8fdbed16810f7379a \
|
||||
&& patch -l -p1 <<EOF
|
||||
diff --git a/acorn-loose/rollup.config.js b/acorn-loose/rollup.config.js
|
||||
index d2389b2..c37882b 100644
|
||||
--- a/acorn-loose/rollup.config.js
|
||||
+++ b/acorn-loose/rollup.config.js
|
||||
@@ -1,2 +0,0 @@
|
||||
-import buble from "rollup-plugin-buble"
|
||||
-
|
||||
@@ -23 +20,0 @@ export default {
|
||||
- buble({transforms: {dangerousForOf: true}})
|
||||
diff --git a/acorn-walk/rollup.config.js b/acorn-walk/rollup.config.js
|
||||
index 67dd613..8c28807 100644
|
||||
--- a/acorn-walk/rollup.config.js
|
||||
+++ b/acorn-walk/rollup.config.js
|
||||
@@ -1,2 +0,0 @@
|
||||
-import buble from "rollup-plugin-buble"
|
||||
-
|
||||
@@ -19 +16,0 @@ export default {
|
||||
- buble({transforms: {dangerousForOf: true}})
|
||||
diff --git a/acorn/rollup.config.bin.js b/acorn/rollup.config.bin.js
|
||||
index 8a082b0..b3eda60 100644
|
||||
--- a/acorn/rollup.config.bin.js
|
||||
+++ b/acorn/rollup.config.bin.js
|
||||
@@ -1,2 +0,0 @@
|
||||
-import buble from "rollup-plugin-buble"
|
||||
-
|
||||
@@ -11 +9 @@ export default {
|
||||
- plugins: [buble()]
|
||||
+ plugins: []
|
||||
diff --git a/acorn/rollup.config.js b/acorn/rollup.config.js
|
||||
index c775a0c..cfd4c68 100644
|
||||
--- a/acorn/rollup.config.js
|
||||
+++ b/acorn/rollup.config.js
|
||||
@@ -1,2 +0,0 @@
|
||||
-import buble from "rollup-plugin-buble"
|
||||
-
|
||||
@@ -19 +16,0 @@ export default {
|
||||
- buble({transforms: {dangerousForOf: true}})
|
||||
diff --git a/package.json b/package.json
|
||||
index 382f59e..4612a75 100644
|
||||
--- a/package.json
|
||||
+++ b/package.json
|
||||
@@ -24,4 +24 @@
|
||||
- "prepare": "npm run test",
|
||||
- "test": "node test/run.js && node test/lint.js",
|
||||
- "pretest": "npm run build:main && npm run build:loose",
|
||||
- "test:test262": "node bin/run_test262.js",
|
||||
+ "test": "node test/run.js",
|
||||
@@ -32,2 +29 @@
|
||||
- "build:bin": "rollup -c acorn/rollup.config.bin.js",
|
||||
- "lint": "eslint acorn/src/ acorn-walk/src/ acorn-loose/src/"
|
||||
+ "build:bin": "rollup -c acorn/rollup.config.bin.js"
|
||||
@@ -36,6 +31,0 @@
|
||||
- "eslint": "^4.10.0",
|
||||
- "eslint-config-standard": "^10.2.1",
|
||||
- "eslint-plugin-import": "^2.2.0",
|
||||
- "eslint-plugin-node": "^5.2.1",
|
||||
- "eslint-plugin-promise": "^3.5.0",
|
||||
- "eslint-plugin-standard": "^3.0.1",
|
||||
@@ -43,4 +32,0 @@
|
||||
- "rollup-plugin-buble": "^0.19.0",
|
||||
- "test262": "git+https://github.com/tc39/test262.git#a6c819ad0f049f23f1a37af6b89dbb79fe3b9216",
|
||||
- "test262-parser-runner": "^0.5.0",
|
||||
- "test262-stream": "^1.2.1",
|
||||
EOF
|
||||
ERR=$?; if [ "$ERR" != "0" ]; then echo "Error: $ERR"; exit $ERR; fi
|
||||
minify_in_situ "acorn/src" \
|
||||
&& minify_in_situ "acorn-loose/src" \
|
||||
&& minify_in_situ "acorn-walk/src" \
|
||||
&& rm -rf node_modules \
|
||||
&& npm_install \
|
||||
&& rm -rf acorn/dist acorn-loose/dist acorn-walk/dist \
|
||||
&& npm run build \
|
||||
&& minify_in_situ "acorn/dist" \
|
||||
&& minify_in_situ "acorn-loose/dist" \
|
||||
&& minify_in_situ "acorn-walk/dist" \
|
||||
&& npm test
|
||||
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
|
||||
52
test/release/buble.sh
Executable file
52
test/release/buble.sh
Executable file
@@ -0,0 +1,52 @@
|
||||
#!/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/buble \
|
||||
&& git clone https://github.com/bublejs/buble.git tmp/buble \
|
||||
&& cd tmp/buble \
|
||||
&& rm -rf .git/hooks \
|
||||
&& git checkout dcc5ab02c9af6ddaad94e587c4911677340ec100 \
|
||||
&& patch -l -p1 <<EOF
|
||||
--- a/package.json
|
||||
+++ b/package.json
|
||||
@@ -29 +28,0 @@
|
||||
- "prepublish": "npm test",
|
||||
@@ -67,3 +66 @@
|
||||
- "source-map-support": "^0.5.16",
|
||||
- "test262": "git+https://github.com/tc39/test262.git#4f1155c566a222238fd86f179c6635ecb4c289bb",
|
||||
- "test262-stream": "^1.3.0"
|
||||
+ "source-map-support": "^0.5.16"
|
||||
--- a/src/program/BlockStatement.js
|
||||
+++ b/src/program/BlockStatement.js
|
||||
@@ -309 +309 @@ export default class BlockStatement extends Node {
|
||||
- let cont = false; // TODO implement proper continue...
|
||||
+ let cont = !declarations; // TODO implement proper continue...
|
||||
EOF
|
||||
ERR=$?; if [ "$ERR" != "0" ]; then echo "Error: $ERR"; exit $ERR; fi
|
||||
minify_in_situ "src" \
|
||||
&& rm -rf node_modules \
|
||||
&& npm_install \
|
||||
&& rm -rf dist \
|
||||
&& npm run build \
|
||||
&& minify_in_situ "dist" \
|
||||
&& node_modules/.bin/mocha
|
||||
52
test/release/butternut.sh
Executable file
52
test/release/butternut.sh
Executable file
@@ -0,0 +1,52 @@
|
||||
#!/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/butternut \
|
||||
&& git clone https://github.com/Rich-Harris/butternut.git tmp/butternut \
|
||||
&& cd tmp/butternut \
|
||||
&& rm -rf .git/hooks \
|
||||
&& patch -l -p1 <<EOF
|
||||
--- a/package.json
|
||||
+++ b/package.json
|
||||
@@ -25 +24,0 @@
|
||||
- "prepublish": "npm run test:min",
|
||||
--- a/rollup.config.js
|
||||
+++ b/rollup.config.js
|
||||
@@ -1 +0,0 @@
|
||||
-import buble from 'rollup-plugin-buble';
|
||||
@@ -28,6 +26,0 @@ const config = {
|
||||
- buble({
|
||||
- include: ['src/**', 'node_modules/acorn/**'],
|
||||
- transforms: {
|
||||
- dangerousForOf: true
|
||||
- }
|
||||
- }),
|
||||
EOF
|
||||
ERR=$?; if [ "$ERR" != "0" ]; then echo "Error: $ERR"; exit $ERR; fi
|
||||
minify_in_situ "src" \
|
||||
&& rm -rf node_modules \
|
||||
&& npm_install \
|
||||
&& rm -rf dist \
|
||||
&& npm run build \
|
||||
&& minify_in_situ "dist" \
|
||||
&& node_modules/.bin/mocha test/test.js
|
||||
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
|
||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user