Compare commits
103 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
0b0296eb2a | ||
|
|
872270b149 | ||
|
|
b1c593a041 | ||
|
|
13be50a4a9 | ||
|
|
16cd5d57a5 | ||
|
|
834f9f3924 | ||
|
|
cf0951f726 | ||
|
|
852f78491a | ||
|
|
229e42cdee | ||
|
|
4e49302916 | ||
|
|
1e51586996 | ||
|
|
d48a3080ac | ||
|
|
26fbeece1c | ||
|
|
8898b8a0fe | ||
|
|
ec64acd2c8 | ||
|
|
ac0b61ed6e | ||
|
|
c06a50f338 | ||
|
|
09f9ae2de9 | ||
|
|
7e6331bb39 | ||
|
|
e275148998 | ||
|
|
974247c8c0 | ||
|
|
a0f4fd390a | ||
|
|
b8b133d91a | ||
|
|
c525a2b190 | ||
|
|
6ffbecb72b | ||
|
|
f0ff6189be | ||
|
|
6b3c49e458 | ||
|
|
f584ca8d07 | ||
|
|
ae4db00991 | ||
|
|
100307ab31 | ||
|
|
148047fbbf | ||
|
|
d11dca3cf9 | ||
|
|
e5badb9541 | ||
|
|
fa668a28b4 | ||
|
|
686a496b1c | ||
|
|
11676f9d72 | ||
|
|
dd31d12a91 | ||
|
|
eb55d8a9bb | ||
|
|
81f1df14d7 | ||
|
|
7f8d72d9d3 | ||
|
|
1eaa211e09 | ||
|
|
0610c020b1 | ||
|
|
0d7d4918eb | ||
|
|
48284844a4 | ||
|
|
ec2e5fa3a2 | ||
|
|
da17766ddd | ||
|
|
0913db8c84 | ||
|
|
5c7705fcad | ||
|
|
f6372483a0 | ||
|
|
98f330658f | ||
|
|
a7b3b0d3a5 | ||
|
|
0a35acbbe7 | ||
|
|
2a9989dd18 | ||
|
|
79b98a9fe8 | ||
|
|
057de570e6 | ||
|
|
557b3e412f | ||
|
|
8d74f34373 | ||
|
|
266ddd9639 | ||
|
|
e51c6ba380 | ||
|
|
6389e52305 | ||
|
|
e05510f3bc | ||
|
|
fc9804b909 | ||
|
|
4761d07e0b | ||
|
|
0111497fc9 | ||
|
|
7d8dea3b26 | ||
|
|
25fc02743a | ||
|
|
0bd8053524 | ||
|
|
1a78bbcd23 | ||
|
|
8430123e9d | ||
|
|
614db97cca | ||
|
|
d854523783 | ||
|
|
781f26eda1 | ||
|
|
37f4395cc0 | ||
|
|
de619ae5a6 | ||
|
|
86859f6d7e | ||
|
|
dcdcfe4d39 | ||
|
|
72306b9885 | ||
|
|
38756b1f26 | ||
|
|
85a09fc3b6 | ||
|
|
307b88d6cc | ||
|
|
fb049d3a81 | ||
|
|
67cca43358 | ||
|
|
642273c290 | ||
|
|
e8b23c7798 | ||
|
|
9edbe93df5 | ||
|
|
af37ecafe1 | ||
|
|
41a9329409 | ||
|
|
7eb52d2837 | ||
|
|
eb63fece2f | ||
|
|
2d8af8947e | ||
|
|
2650182f47 | ||
|
|
572b97b0bb | ||
|
|
698705a820 | ||
|
|
debc525fa1 | ||
|
|
5576e2737a | ||
|
|
b40d5de69c | ||
|
|
b7ef7840f3 | ||
|
|
85924bb32e | ||
|
|
a97690fc72 | ||
|
|
02c638209e | ||
|
|
030611b729 | ||
|
|
335b72df03 | ||
|
|
3a7d53f3cf |
1
.gitattributes
vendored
Normal file
1
.gitattributes
vendored
Normal file
@@ -0,0 +1 @@
|
|||||||
|
*.js text eol=lf
|
||||||
194
README.md
194
README.md
@@ -10,6 +10,9 @@ There's also an
|
|||||||
[in-browser online demo](http://lisperator.net/uglifyjs/#demo) (for Firefox,
|
[in-browser online demo](http://lisperator.net/uglifyjs/#demo) (for Firefox,
|
||||||
Chrome and probably Safari).
|
Chrome and probably Safari).
|
||||||
|
|
||||||
|
Note: release versions of `uglify-js` only support ECMAScript 5 (ES5). If you wish to minify
|
||||||
|
ES2015+ (ES6+) code then please use the [harmony](#harmony) development branch.
|
||||||
|
|
||||||
Install
|
Install
|
||||||
-------
|
-------
|
||||||
|
|
||||||
@@ -62,12 +65,18 @@ The available options are:
|
|||||||
--source-map-include-sources Pass this flag if you want to include the
|
--source-map-include-sources Pass this flag if you want to include the
|
||||||
content of source files in the source map as
|
content of source files in the source map as
|
||||||
sourcesContent property.
|
sourcesContent property.
|
||||||
|
--source-map-inline Write base64-encoded source map to the end of js output.
|
||||||
--in-source-map Input source map, useful if you're compressing
|
--in-source-map Input source map, useful if you're compressing
|
||||||
JS that was generated from some other original
|
JS that was generated from some other original
|
||||||
code.
|
code.
|
||||||
--screw-ie8 Pass this flag if you don't care about full
|
--screw-ie8 Use this flag if you don't wish to support
|
||||||
compliance with Internet Explorer 6-8 quirks
|
Internet Explorer 6/7/8.
|
||||||
(by default UglifyJS will try to be IE-proof).
|
By default UglifyJS will not try to be IE-proof.
|
||||||
|
--support-ie8 Use this flag to support Internet Explorer 6/7/8.
|
||||||
|
Equivalent to setting `screw_ie8: false` in `minify()`
|
||||||
|
for `compress`, `mangle` and `output` options.
|
||||||
|
Note: `--support-ie8` may generate incorrect code
|
||||||
|
for `try`/`catch` in ES5 compliant browsers.
|
||||||
--expr Parse a single expression, rather than a
|
--expr Parse a single expression, rather than a
|
||||||
program (for parsing JSON)
|
program (for parsing JSON)
|
||||||
-p, --prefix Skip prefix for original filenames that appear
|
-p, --prefix Skip prefix for original filenames that appear
|
||||||
@@ -81,10 +90,9 @@ The available options are:
|
|||||||
-b, --beautify Beautify output/specify output options.
|
-b, --beautify Beautify output/specify output options.
|
||||||
-m, --mangle Mangle names/pass mangler options.
|
-m, --mangle Mangle names/pass mangler options.
|
||||||
-r, --reserved Reserved names to exclude from mangling.
|
-r, --reserved Reserved names to exclude from mangling.
|
||||||
-c, --compress Enable compressor/pass compressor options. Pass
|
-c, --compress Enable compressor/pass compressor options, e.g.
|
||||||
options like -c
|
`-c 'if_return=false,pure_funcs=["Math.pow","console.log"]'`
|
||||||
hoist_vars=false,if_return=false. Use -c with
|
Use `-c` with no argument to enable default compression
|
||||||
no argument to use the default compression
|
|
||||||
options.
|
options.
|
||||||
-d, --define Global definitions
|
-d, --define Global definitions
|
||||||
-e, --enclose Embed everything in a big function, with a
|
-e, --enclose Embed everything in a big function, with a
|
||||||
@@ -95,8 +103,8 @@ The available options are:
|
|||||||
"@preserve". You can optionally pass one of the
|
"@preserve". You can optionally pass one of the
|
||||||
following arguments to this flag:
|
following arguments to this flag:
|
||||||
- "all" to keep all comments
|
- "all" to keep all comments
|
||||||
- a valid JS regexp (needs to start with a
|
- a valid JS RegExp like `/foo/` or `/^!/` to
|
||||||
slash) to keep only comments that match.
|
keep only matching comments.
|
||||||
Note that currently not *all* comments can be
|
Note that currently not *all* comments can be
|
||||||
kept when compression is on, because of dead
|
kept when compression is on, because of dead
|
||||||
code removal or cascading statements into
|
code removal or cascading statements into
|
||||||
@@ -133,11 +141,11 @@ The available options are:
|
|||||||
--reserved-file File containing reserved names
|
--reserved-file File containing reserved names
|
||||||
--reserve-domprops Make (most?) DOM properties reserved for
|
--reserve-domprops Make (most?) DOM properties reserved for
|
||||||
--mangle-props
|
--mangle-props
|
||||||
--mangle-props Mangle property names (default `0`). Set to
|
--mangle-props Mangle property names (default `0`). Set to
|
||||||
`true` or `1` to mangle all property names. Set
|
`true` or `1` to mangle all property names. Set
|
||||||
to `unquoted` or `2` to only mangle unquoted
|
to `unquoted` or `2` to only mangle unquoted
|
||||||
property names. Mode `2` also enables the
|
property names. Mode `2` also enables the
|
||||||
`keep_quoted_props` beautifier option to
|
`keep_quoted_props` beautifier option to
|
||||||
preserve the quotes around property names and
|
preserve the quotes around property names and
|
||||||
disables the `properties` compressor option to
|
disables the `properties` compressor option to
|
||||||
prevent rewriting quoted properties with dot
|
prevent rewriting quoted properties with dot
|
||||||
@@ -145,8 +153,10 @@ The available options are:
|
|||||||
them explicitly on the command line.
|
them explicitly on the command line.
|
||||||
--mangle-regex Only mangle property names matching the regex
|
--mangle-regex Only mangle property names matching the regex
|
||||||
--name-cache File to hold mangled names mappings
|
--name-cache File to hold mangled names mappings
|
||||||
--pure-funcs List of functions that can be safely removed if
|
--pure-funcs Functions that can be safely removed if their
|
||||||
their return value is not used [array]
|
return value is not used, e.g.
|
||||||
|
`--pure-funcs Math.floor console.info`
|
||||||
|
(requires `--compress`)
|
||||||
```
|
```
|
||||||
|
|
||||||
Specify `--output` (`-o`) to declare the output file. Otherwise the output
|
Specify `--output` (`-o`) to declare the output file. Otherwise the output
|
||||||
@@ -282,6 +292,32 @@ of mangled property names.
|
|||||||
Using the name cache is not necessary if you compress all your files in a
|
Using the name cache is not necessary if you compress all your files in a
|
||||||
single call to UglifyJS.
|
single call to UglifyJS.
|
||||||
|
|
||||||
|
#### Mangling unquoted names (`--mangle-props=unquoted` or `--mangle-props=2`)
|
||||||
|
|
||||||
|
Using quoted property name (`o["foo"]`) reserves the property name (`foo`)
|
||||||
|
so that it is not mangled throughout the entire script even when used in an
|
||||||
|
unquoted style (`o.foo`). Example:
|
||||||
|
|
||||||
|
```
|
||||||
|
$ echo 'var o={"foo":1, bar:3}; o.foo += o.bar; console.log(o.foo);' | uglifyjs --mangle-props=2 -mc
|
||||||
|
var o={"foo":1,a:3};o.foo+=o.a,console.log(o.foo);
|
||||||
|
```
|
||||||
|
|
||||||
|
#### Debugging property name mangling
|
||||||
|
|
||||||
|
You can also pass `--mangle-props-debug` in order to mangle property names
|
||||||
|
without completely obscuring them. For example the property `o.foo`
|
||||||
|
would mangle to `o._$foo$_` with this option. This allows property mangling
|
||||||
|
of a large codebase while still being able to debug the code and identify
|
||||||
|
where mangling is breaking things.
|
||||||
|
|
||||||
|
You can also pass a custom suffix using `--mangle-props-debug=XYZ`. This would then
|
||||||
|
mangle `o.foo` to `o._$foo$XYZ_`. You can change this each time you compile a
|
||||||
|
script to identify how a property got mangled. One technique is to pass a
|
||||||
|
random number on every compile to simulate mangling changing with different
|
||||||
|
inputs (e.g. as you update the input script with new properties), and to help
|
||||||
|
identify mistakes like writing mangled keys to storage.
|
||||||
|
|
||||||
## Compressor options
|
## Compressor options
|
||||||
|
|
||||||
You need to pass `--compress` (`-c`) to enable the compressor. Optionally
|
You need to pass `--compress` (`-c`) to enable the compressor. Optionally
|
||||||
@@ -289,7 +325,14 @@ you can pass a comma-separated list of options. Options are in the form
|
|||||||
`foo=bar`, or just `foo` (the latter implies a boolean option that you want
|
`foo=bar`, or just `foo` (the latter implies a boolean option that you want
|
||||||
to set `true`; it's effectively a shortcut for `foo=true`).
|
to set `true`; it's effectively a shortcut for `foo=true`).
|
||||||
|
|
||||||
- `sequences` -- join consecutive simple statements using the comma operator
|
- `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`
|
||||||
|
to disable. The smallest `sequences` length is `2`. A `sequences` value of `1`
|
||||||
|
is grandfathered to be equivalent to `true` and as such means `200`. On rare
|
||||||
|
occasions the default sequences limit leads to very slow compress times in which
|
||||||
|
case a value of `20` or less is recommended.
|
||||||
|
|
||||||
- `properties` -- rewrite property access using the dot notation, for
|
- `properties` -- rewrite property access using the dot notation, for
|
||||||
example `foo["bar"] → foo.bar`
|
example `foo["bar"] → foo.bar`
|
||||||
@@ -307,6 +350,9 @@ to set `true`; it's effectively a shortcut for `foo=true`).
|
|||||||
comparison are switching. Compression only works if both `comparisons` and
|
comparison are switching. Compression only works if both `comparisons` and
|
||||||
`unsafe_comps` are both set to true.
|
`unsafe_comps` are both set to true.
|
||||||
|
|
||||||
|
- `unsafe_proto` (default: false) -- optimize expressions like
|
||||||
|
`Array.prototype.slice.call(a)` into `[].slice.call(a)`
|
||||||
|
|
||||||
- `conditionals` -- apply optimizations for `if`-s and conditional
|
- `conditionals` -- apply optimizations for `if`-s and conditional
|
||||||
expressions
|
expressions
|
||||||
|
|
||||||
@@ -322,7 +368,15 @@ to set `true`; it's effectively a shortcut for `foo=true`).
|
|||||||
- `loops` -- optimizations for `do`, `while` and `for` loops when we can
|
- `loops` -- optimizations for `do`, `while` and `for` loops when we can
|
||||||
statically determine the condition
|
statically determine the condition
|
||||||
|
|
||||||
- `unused` -- drop unreferenced functions and variables
|
- `unused` -- drop unreferenced functions and variables (simple direct variable
|
||||||
|
assignments do not count as references unless set to `"keep_assign"`)
|
||||||
|
|
||||||
|
- `toplevel` -- drop unreferenced functions (`"funcs"`) and/or variables (`"vars"`)
|
||||||
|
in the toplevel scope (`false` by default, `true` to drop both unreferenced
|
||||||
|
functions and variables)
|
||||||
|
|
||||||
|
- `top_retain` -- prevent specific toplevel functions and variables from `unused`
|
||||||
|
removal (can be array, comma-separated, RegExp or function. Implies `toplevel`)
|
||||||
|
|
||||||
- `hoist_funs` -- hoist function declarations
|
- `hoist_funs` -- hoist function declarations
|
||||||
|
|
||||||
@@ -339,6 +393,9 @@ to set `true`; it's effectively a shortcut for `foo=true`).
|
|||||||
- `collapse_vars` -- default `false`. Collapse single-use `var` and `const`
|
- `collapse_vars` -- default `false`. Collapse single-use `var` and `const`
|
||||||
definitions when possible.
|
definitions when possible.
|
||||||
|
|
||||||
|
- `reduce_vars` -- default `false`. Improve optimization on variables assigned
|
||||||
|
with and used as constant values.
|
||||||
|
|
||||||
- `warnings` -- display warnings when dropping unreachable code or unused
|
- `warnings` -- display warnings when dropping unreachable code or unused
|
||||||
declarations etc.
|
declarations etc.
|
||||||
|
|
||||||
@@ -362,15 +419,17 @@ to set `true`; it's effectively a shortcut for `foo=true`).
|
|||||||
overhead (compression will be slower).
|
overhead (compression will be slower).
|
||||||
|
|
||||||
- `drop_console` -- default `false`. Pass `true` to discard calls to
|
- `drop_console` -- default `false`. Pass `true` to discard calls to
|
||||||
`console.*` functions.
|
`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.
|
||||||
|
|
||||||
- `keep_fargs` -- default `true`. Prevents the
|
- `keep_fargs` -- default `true`. Prevents the
|
||||||
compressor from discarding unused function arguments. You need this
|
compressor from discarding unused function arguments. You need this
|
||||||
for code which relies on `Function.length`.
|
for code which relies on `Function.length`.
|
||||||
|
|
||||||
- `keep_fnames` -- default `false`. Pass `true` to prevent the
|
- `keep_fnames` -- default `false`. Pass `true` to prevent the
|
||||||
compressor from mangling/discarding function names. Useful for code relying on
|
compressor from discarding function names. Useful for code relying on
|
||||||
`Function.prototype.name`.
|
`Function.prototype.name`. See also: the `keep_fnames` [mangle option](#mangle).
|
||||||
|
|
||||||
- `passes` -- default `1`. Number of times to run compress. Use an
|
- `passes` -- default `1`. Number of times to run compress. Use an
|
||||||
integer argument larger than 1 to further reduce code size in some cases.
|
integer argument larger than 1 to further reduce code size in some cases.
|
||||||
@@ -404,6 +463,8 @@ if (DEBUG) {
|
|||||||
}
|
}
|
||||||
```
|
```
|
||||||
|
|
||||||
|
You can specify nested constants in the form of `--define env.DEBUG=false`.
|
||||||
|
|
||||||
UglifyJS will warn about the condition being always false and about dropping
|
UglifyJS will warn about the condition being always false and about dropping
|
||||||
unreachable code; for now there is no option to turn off only this specific
|
unreachable code; for now there is no option to turn off only this specific
|
||||||
warning, you can pass `warnings=false` to turn off *all* warnings.
|
warning, you can pass `warnings=false` to turn off *all* warnings.
|
||||||
@@ -414,8 +475,6 @@ separate file and include it into the build. For example you can have a
|
|||||||
```javascript
|
```javascript
|
||||||
const DEBUG = false;
|
const DEBUG = false;
|
||||||
const PRODUCTION = true;
|
const PRODUCTION = true;
|
||||||
// Alternative for environments that don't support `const`
|
|
||||||
/** @const */ var STAGING = false;
|
|
||||||
// etc.
|
// etc.
|
||||||
```
|
```
|
||||||
|
|
||||||
@@ -426,7 +485,8 @@ and build your code like this:
|
|||||||
UglifyJS will notice the constants and, since they cannot be altered, it
|
UglifyJS will notice the constants and, since they cannot be altered, it
|
||||||
will evaluate references to them to the value itself and drop unreachable
|
will evaluate references to them to the value itself and drop unreachable
|
||||||
code as usual. The build will contain the `const` declarations if you use
|
code as usual. The build will contain the `const` declarations if you use
|
||||||
them. If you are targeting < ES6 environments, use `/** @const */ var`.
|
them. If you are targeting < ES6 environments which does not support `const`,
|
||||||
|
using `var` with `reduce_vars` (enabled by default) should suffice.
|
||||||
|
|
||||||
<a name="codegen-options"></a>
|
<a name="codegen-options"></a>
|
||||||
|
|
||||||
@@ -619,9 +679,26 @@ console.log(result.code); // minified output
|
|||||||
console.log(result.map);
|
console.log(result.map);
|
||||||
```
|
```
|
||||||
|
|
||||||
|
To generate a source map with the fromString option, you can also use an object:
|
||||||
|
```javascript
|
||||||
|
var result = UglifyJS.minify({"file1.js": "var a = function () {};"}, {
|
||||||
|
outSourceMap: "out.js.map",
|
||||||
|
outFileName: "out.js",
|
||||||
|
fromString: true
|
||||||
|
});
|
||||||
|
```
|
||||||
|
|
||||||
Note that the source map is not saved in a file, it's just returned in
|
Note that the source map is not saved in a file, it's just returned in
|
||||||
`result.map`. The value passed for `outSourceMap` is only used to set the
|
`result.map`. The value passed for `outSourceMap` is only used to set
|
||||||
`file` attribute in the source map (see [the spec][sm-spec]).
|
`//# sourceMappingURL=out.js.map` in `result.code`. The value of
|
||||||
|
`outFileName` is only used to set `file` attribute in source map file.
|
||||||
|
|
||||||
|
The `file` attribute in the source map (see [the spec][sm-spec]) will
|
||||||
|
use `outFileName` firstly, if it's falsy, then will be deduced from
|
||||||
|
`outSourceMap` (by removing `'.map'`).
|
||||||
|
|
||||||
|
You can set option `sourceMapInline` to be `true` and source map will
|
||||||
|
be appended to code.
|
||||||
|
|
||||||
You can also specify sourceRoot property to be included in source map:
|
You can also specify sourceRoot property to be included in source map:
|
||||||
```javascript
|
```javascript
|
||||||
@@ -654,6 +731,17 @@ var result = UglifyJS.minify("compiled.js", {
|
|||||||
The `inSourceMap` is only used if you also request `outSourceMap` (it makes
|
The `inSourceMap` is only used if you also request `outSourceMap` (it makes
|
||||||
no sense otherwise).
|
no sense otherwise).
|
||||||
|
|
||||||
|
To set the source map url, use the `sourceMapUrl` option.
|
||||||
|
If you're using the X-SourceMap header instead, you can just set the `sourceMapUrl` option to false.
|
||||||
|
Defaults to outSourceMap:
|
||||||
|
|
||||||
|
```javascript
|
||||||
|
var result = UglifyJS.minify([ "file1.js" ], {
|
||||||
|
outSourceMap: "out.js.map",
|
||||||
|
sourceMapUrl: "localhost/out.js.map"
|
||||||
|
});
|
||||||
|
```
|
||||||
|
|
||||||
Other options:
|
Other options:
|
||||||
|
|
||||||
- `warnings` (default `false`) — pass `true` to display compressor warnings.
|
- `warnings` (default `false`) — pass `true` to display compressor warnings.
|
||||||
@@ -661,7 +749,8 @@ Other options:
|
|||||||
- `fromString` (default `false`) — if you pass `true` then you can pass
|
- `fromString` (default `false`) — if you pass `true` then you can pass
|
||||||
JavaScript source code, rather than file names.
|
JavaScript source code, rather than file names.
|
||||||
|
|
||||||
- `mangle` — pass `false` to skip mangling names.
|
- `mangle` (default `true`) — pass `false` to skip mangling names, or pass
|
||||||
|
an object to specify mangling options (see below).
|
||||||
|
|
||||||
- `mangleProperties` (default `false`) — pass an object to specify custom
|
- `mangleProperties` (default `false`) — pass an object to specify custom
|
||||||
mangle property options.
|
mangle property options.
|
||||||
@@ -680,10 +769,41 @@ Other options:
|
|||||||
|
|
||||||
- `except` - pass an array of identifiers that should be excluded from mangling
|
- `except` - pass an array of identifiers that should be excluded from mangling
|
||||||
|
|
||||||
|
- `toplevel` — mangle names declared in the toplevel scope (disabled by
|
||||||
|
default).
|
||||||
|
|
||||||
|
- `eval` — mangle names visible in scopes where eval or with are used
|
||||||
|
(disabled by default).
|
||||||
|
|
||||||
|
- `keep_fnames` -- default `false`. Pass `true` to not mangle
|
||||||
|
function names. Useful for code relying on `Function.prototype.name`.
|
||||||
|
See also: the `keep_fnames` [compress option](#compressor-options).
|
||||||
|
|
||||||
|
Examples:
|
||||||
|
|
||||||
|
```javascript
|
||||||
|
//tst.js
|
||||||
|
var globalVar;
|
||||||
|
function funcName(firstLongName, anotherLongName)
|
||||||
|
{
|
||||||
|
var myVariable = firstLongName + anotherLongName;
|
||||||
|
}
|
||||||
|
|
||||||
|
UglifyJS.minify("tst.js").code;
|
||||||
|
// 'function funcName(a,n){}var globalVar;'
|
||||||
|
|
||||||
|
UglifyJS.minify("tst.js", { mangle: { except: ['firstLongName'] } }).code;
|
||||||
|
// 'function funcName(firstLongName,a){}var globalVar;'
|
||||||
|
|
||||||
|
UglifyJS.minify("tst.js", { mangle: { toplevel: true } }).code;
|
||||||
|
// 'function n(n,a){}var a;'
|
||||||
|
```
|
||||||
|
|
||||||
##### mangleProperties options
|
##### mangleProperties options
|
||||||
|
|
||||||
- `regex` — Pass a RegExp to only mangle certain names (maps to the `--mangle-regex` CLI arguments option)
|
- `regex` — Pass a RegExp to only mangle certain names (maps to the `--mangle-regex` CLI arguments option)
|
||||||
- `ignore_quoted` – Only mangle unquoted property names (maps to the `--mangle-props 2` CLI arguments option)
|
- `ignore_quoted` – Only mangle unquoted property names (maps to the `--mangle-props 2` CLI arguments option)
|
||||||
|
- `debug` – Mangle names with the original name still present (maps to the `--mangle-props-debug` CLI arguments option). Defaults to `false`. Pass an empty string to enable, or a non-empty string to set the suffix.
|
||||||
|
|
||||||
We could add more options to `UglifyJS.minify` — if you need additional
|
We could add more options to `UglifyJS.minify` — if you need additional
|
||||||
functionality please suggest!
|
functionality please suggest!
|
||||||
@@ -790,8 +910,11 @@ which we care about here are `source_map` and `comments`.
|
|||||||
#### Keeping comments in the output
|
#### Keeping comments in the output
|
||||||
|
|
||||||
In order to keep certain comments in the output you need to pass the
|
In order to keep certain comments in the output you need to pass the
|
||||||
`comments` option. Pass a RegExp or a function. If you pass a RegExp, only
|
`comments` option. Pass a RegExp (as string starting and closing with `/`
|
||||||
those comments whose body matches the regexp will be kept. Note that body
|
or pass a RegExp object), a boolean or a function. Stringified options
|
||||||
|
`all` and `some` can be passed too, where `some` behaves like it's cli
|
||||||
|
equivalent `--comments` without passing a value. If you pass a RegExp,
|
||||||
|
only those comments whose body matches the RegExp will be kept. Note that body
|
||||||
means without the initial `//` or `/*`. If you pass a function, it will be
|
means without the initial `//` or `/*`. If you pass a function, it will be
|
||||||
called for every comment in the tree and will receive two arguments: the
|
called for every comment in the tree and will receive two arguments: the
|
||||||
node that the comment is attached to, and the comment token itself.
|
node that the comment is attached to, and the comment token itself.
|
||||||
@@ -846,3 +969,20 @@ The `source_map_options` (optional) can contain the following properties:
|
|||||||
[codegen]: http://lisperator.net/uglifyjs/codegen
|
[codegen]: http://lisperator.net/uglifyjs/codegen
|
||||||
[compressor]: http://lisperator.net/uglifyjs/compress
|
[compressor]: http://lisperator.net/uglifyjs/compress
|
||||||
[parser]: http://lisperator.net/uglifyjs/parser
|
[parser]: http://lisperator.net/uglifyjs/parser
|
||||||
|
|
||||||
|
#### Harmony
|
||||||
|
|
||||||
|
If you wish to use the experimental [harmony](https://github.com/mishoo/UglifyJS2/commits/harmony)
|
||||||
|
branch to minify ES2015+ (ES6+) code please use the following in your `package.json` file:
|
||||||
|
|
||||||
|
```
|
||||||
|
"uglify-js": "git+https://github.com/mishoo/UglifyJS2.git#harmony"
|
||||||
|
```
|
||||||
|
|
||||||
|
or to directly install the experimental harmony version of uglify:
|
||||||
|
|
||||||
|
```
|
||||||
|
npm install --save-dev uglify-js@github:mishoo/UglifyJS2#harmony
|
||||||
|
```
|
||||||
|
|
||||||
|
See [#448](https://github.com/mishoo/UglifyJS2/issues/448) for additional details.
|
||||||
|
|||||||
24
appveyor.yml
Normal file
24
appveyor.yml
Normal file
@@ -0,0 +1,24 @@
|
|||||||
|
environment:
|
||||||
|
matrix:
|
||||||
|
- nodejs_version: "0.10"
|
||||||
|
- nodejs_version: "0.12"
|
||||||
|
- nodejs_version: "4.0"
|
||||||
|
- nodejs_version: "6.0"
|
||||||
|
|
||||||
|
matrix:
|
||||||
|
fast_finish: true
|
||||||
|
|
||||||
|
platform:
|
||||||
|
- x86
|
||||||
|
- x64
|
||||||
|
|
||||||
|
install:
|
||||||
|
- ps: Install-Product node $env:nodejs_version $env:platform
|
||||||
|
- npm install
|
||||||
|
|
||||||
|
test_script:
|
||||||
|
- node --version
|
||||||
|
- npm --version
|
||||||
|
- npm test
|
||||||
|
|
||||||
|
build: off
|
||||||
180
bin/uglifyjs
180
bin/uglifyjs
@@ -10,6 +10,7 @@ var fs = require("fs");
|
|||||||
var path = require("path");
|
var path = require("path");
|
||||||
var async = require("async");
|
var async = require("async");
|
||||||
var acorn;
|
var acorn;
|
||||||
|
var screw_ie8 = true;
|
||||||
var ARGS = yargs
|
var ARGS = yargs
|
||||||
.usage("$0 input1.js [input2.js ...] [options]\n\
|
.usage("$0 input1.js [input2.js ...] [options]\n\
|
||||||
Use a single dash to read input from the standard input.\
|
Use a single dash to read input from the standard input.\
|
||||||
@@ -22,9 +23,11 @@ mangling you need to use `-c` and `-m`.\
|
|||||||
.describe("source-map", "Specify an output file where to generate source map.")
|
.describe("source-map", "Specify an output file where to generate source map.")
|
||||||
.describe("source-map-root", "The path to the original source to be included in the source map.")
|
.describe("source-map-root", "The path to the original source to be included in the source map.")
|
||||||
.describe("source-map-url", "The path to the source map to be added in //# sourceMappingURL. Defaults to the value passed with --source-map.")
|
.describe("source-map-url", "The path to the source map to be added in //# sourceMappingURL. Defaults to the value passed with --source-map.")
|
||||||
|
.describe("source-map-inline", "Write base64-encoded source map to the end of js output. Disabled by default")
|
||||||
.describe("source-map-include-sources", "Pass this flag if you want to include the content of source files in the source map as sourcesContent property.")
|
.describe("source-map-include-sources", "Pass this flag if you want to include the content of source files in the source map as sourcesContent property.")
|
||||||
.describe("in-source-map", "Input source map, useful if you're compressing JS that was generated from some other original code.")
|
.describe("in-source-map", "Input source map, useful if you're compressing JS that was generated from some other original code.")
|
||||||
.describe("screw-ie8", "Pass this flag if you don't care about full compliance with Internet Explorer 6-8 quirks (by default UglifyJS will try to be IE-proof).")
|
.describe("screw-ie8", "Do not support Internet Explorer 6/7/8. This flag is enabled by default.")
|
||||||
|
.describe("support-ie8", "Support non-standard Internet Explorer 6/7/8 javascript. Note: may generate incorrect code for try/catch in ES5 compliant browsers.")
|
||||||
.describe("expr", "Parse a single expression, rather than a program (for parsing JSON)")
|
.describe("expr", "Parse a single expression, rather than a program (for parsing JSON)")
|
||||||
.describe("p", "Skip prefix for original filenames that appear in source maps. \
|
.describe("p", "Skip prefix for original filenames that appear in source maps. \
|
||||||
For example -p 3 will drop 3 directories from file names and ensure they are relative paths. \
|
For example -p 3 will drop 3 directories from file names and ensure they are relative paths. \
|
||||||
@@ -44,7 +47,7 @@ Use -c with no argument to use the default compression options.")
|
|||||||
By default this works like Google Closure, keeping JSDoc-style comments that contain \"@license\" or \"@preserve\". \
|
By default this works like Google Closure, keeping JSDoc-style comments that contain \"@license\" or \"@preserve\". \
|
||||||
You can optionally pass one of the following arguments to this flag:\n\
|
You can optionally pass one of the following arguments to this flag:\n\
|
||||||
- \"all\" to keep all comments\n\
|
- \"all\" to keep all comments\n\
|
||||||
- a valid JS regexp (needs to start with a slash) to keep only comments that match.\n\
|
- a valid JS RegExp like `/foo/`or `/^!/` to keep only matching comments.\n\
|
||||||
\
|
\
|
||||||
Note that currently not *all* comments can be kept when compression is on, \
|
Note that currently not *all* comments can be kept when compression is on, \
|
||||||
because of dead code removal or cascading statements into sequences.")
|
because of dead code removal or cascading statements into sequences.")
|
||||||
@@ -74,6 +77,7 @@ You need to pass an argument to this option to specify the name that your module
|
|||||||
.describe("name-cache", "File to hold mangled names mappings")
|
.describe("name-cache", "File to hold mangled names mappings")
|
||||||
.describe("pure-funcs", "List of functions that can be safely removed if their return value is not used")
|
.describe("pure-funcs", "List of functions that can be safely removed if their return value is not used")
|
||||||
.describe("dump-spidermonkey-ast", "Dump SpiderMonkey AST to stdout.")
|
.describe("dump-spidermonkey-ast", "Dump SpiderMonkey AST to stdout.")
|
||||||
|
.describe("wrap-iife", "Wrap IIFEs in parenthesis. Note: this disables the negate_iife compression option")
|
||||||
|
|
||||||
.alias("p", "prefix")
|
.alias("p", "prefix")
|
||||||
.alias("o", "output")
|
.alias("o", "output")
|
||||||
@@ -94,6 +98,7 @@ You need to pass an argument to this option to specify the name that your module
|
|||||||
.string("beautify")
|
.string("beautify")
|
||||||
.string("m")
|
.string("m")
|
||||||
.string("mangle")
|
.string("mangle")
|
||||||
|
.string("mangle-props-debug")
|
||||||
.string("c")
|
.string("c")
|
||||||
.string("compress")
|
.string("compress")
|
||||||
.string("d")
|
.string("d")
|
||||||
@@ -105,12 +110,15 @@ You need to pass an argument to this option to specify the name that your module
|
|||||||
.string("p")
|
.string("p")
|
||||||
.string("prefix")
|
.string("prefix")
|
||||||
.string("name-cache")
|
.string("name-cache")
|
||||||
|
|
||||||
.array("reserved-file")
|
.array("reserved-file")
|
||||||
.array("pure-funcs")
|
.array("pure-funcs")
|
||||||
|
|
||||||
.boolean("expr")
|
.boolean("expr")
|
||||||
|
.boolean("source-map-inline")
|
||||||
.boolean("source-map-include-sources")
|
.boolean("source-map-include-sources")
|
||||||
.boolean("screw-ie8")
|
.boolean("screw-ie8")
|
||||||
|
.boolean("support-ie8")
|
||||||
.boolean("export-all")
|
.boolean("export-all")
|
||||||
.boolean("self")
|
.boolean("self")
|
||||||
.boolean("v")
|
.boolean("v")
|
||||||
@@ -126,6 +134,7 @@ You need to pass an argument to this option to specify the name that your module
|
|||||||
.boolean("bare-returns")
|
.boolean("bare-returns")
|
||||||
.boolean("keep-fnames")
|
.boolean("keep-fnames")
|
||||||
.boolean("reserve-domprops")
|
.boolean("reserve-domprops")
|
||||||
|
.boolean("wrap-iife")
|
||||||
|
|
||||||
.wrap(80)
|
.wrap(80)
|
||||||
|
|
||||||
@@ -219,9 +228,10 @@ if (ARGS.mangle_props === true) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
var OUTPUT_OPTIONS = {
|
var OUTPUT_OPTIONS = {
|
||||||
beautify : BEAUTIFY ? true : false,
|
beautify : BEAUTIFY ? true : false,
|
||||||
preamble : ARGS.preamble || null,
|
max_line_len : 32000,
|
||||||
quote_style : ARGS.quotes != null ? ARGS.quotes : 0
|
preamble : ARGS.preamble || null,
|
||||||
|
quote_style : ARGS.quotes != null ? ARGS.quotes : 0,
|
||||||
};
|
};
|
||||||
|
|
||||||
if (ARGS.mangle_props == 2) {
|
if (ARGS.mangle_props == 2) {
|
||||||
@@ -230,43 +240,38 @@ if (ARGS.mangle_props == 2) {
|
|||||||
COMPRESS.properties = false;
|
COMPRESS.properties = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (ARGS.screw_ie8) {
|
if (ARGS.support_ie8 === true && ARGS.screw_ie8 !== true) {
|
||||||
if (COMPRESS) COMPRESS.screw_ie8 = true;
|
screw_ie8 = false;
|
||||||
if (MANGLE) MANGLE.screw_ie8 = true;
|
|
||||||
OUTPUT_OPTIONS.screw_ie8 = true;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (COMPRESS) COMPRESS.screw_ie8 = screw_ie8;
|
||||||
|
if (MANGLE) MANGLE.screw_ie8 = screw_ie8;
|
||||||
|
OUTPUT_OPTIONS.screw_ie8 = screw_ie8;
|
||||||
|
|
||||||
if (ARGS.keep_fnames) {
|
if (ARGS.keep_fnames) {
|
||||||
if (COMPRESS) COMPRESS.keep_fnames = true;
|
if (COMPRESS) COMPRESS.keep_fnames = true;
|
||||||
if (MANGLE) MANGLE.keep_fnames = true;
|
if (MANGLE) MANGLE.keep_fnames = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (ARGS.wrap_iife) {
|
||||||
|
if (COMPRESS) COMPRESS.negate_iife = false;
|
||||||
|
OUTPUT_OPTIONS.wrap_iife = true;
|
||||||
|
}
|
||||||
|
|
||||||
if (BEAUTIFY)
|
if (BEAUTIFY)
|
||||||
UglifyJS.merge(OUTPUT_OPTIONS, BEAUTIFY);
|
UglifyJS.merge(OUTPUT_OPTIONS, BEAUTIFY);
|
||||||
|
|
||||||
if (ARGS.comments != null) {
|
if (ARGS.comments === "") {
|
||||||
if (/^\/.*\/[a-zA-Z]*$/.test(ARGS.comments)) {
|
OUTPUT_OPTIONS.comments = "some";
|
||||||
try {
|
} else {
|
||||||
OUTPUT_OPTIONS.comments = extractRegex(ARGS.comments);
|
OUTPUT_OPTIONS.comments = ARGS.comments;
|
||||||
} catch (e) {
|
|
||||||
print_error("ERROR: Invalid --comments: " + e.message);
|
|
||||||
}
|
|
||||||
} else if (ARGS.comments == "all") {
|
|
||||||
OUTPUT_OPTIONS.comments = true;
|
|
||||||
} else {
|
|
||||||
OUTPUT_OPTIONS.comments = function(node, comment) {
|
|
||||||
var text = comment.value;
|
|
||||||
var type = comment.type;
|
|
||||||
if (type == "comment2") {
|
|
||||||
// multiline comment
|
|
||||||
return /@preserve|@license|@cc_on/i.test(text);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
var files = ARGS._.slice();
|
var files = ARGS._.slice();
|
||||||
|
|
||||||
|
if (process.platform === "win32")
|
||||||
|
files = UglifyJS.simple_glob(files);
|
||||||
|
|
||||||
if (ARGS.self) {
|
if (ARGS.self) {
|
||||||
if (files.length > 0) {
|
if (files.length > 0) {
|
||||||
print_error("WARN: Ignoring input files since --self was passed");
|
print_error("WARN: Ignoring input files since --self was passed");
|
||||||
@@ -277,21 +282,29 @@ if (ARGS.self) {
|
|||||||
|
|
||||||
var ORIG_MAP = ARGS.in_source_map;
|
var ORIG_MAP = ARGS.in_source_map;
|
||||||
|
|
||||||
if (ORIG_MAP) {
|
if (ORIG_MAP && ORIG_MAP != "inline") {
|
||||||
ORIG_MAP = JSON.parse(fs.readFileSync(ORIG_MAP));
|
ORIG_MAP = JSON.parse(fs.readFileSync(ORIG_MAP));
|
||||||
if (files.length == 0) {
|
if (files.length == 0) {
|
||||||
print_error("INFO: Using file from the input source map: " + ORIG_MAP.file);
|
print_error("INFO: Using file from the input source map: " + ORIG_MAP.file);
|
||||||
files = [ ORIG_MAP.file ];
|
files = [ ORIG_MAP.file ];
|
||||||
}
|
}
|
||||||
if (ARGS.source_map_root == null) {
|
|
||||||
ARGS.source_map_root = ORIG_MAP.sourceRoot;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (files.length == 0) {
|
if (files.length == 0) {
|
||||||
files = [ "-" ];
|
files = [ "-" ];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (ORIG_MAP == "inline") {
|
||||||
|
if (files.length > 1) {
|
||||||
|
print_error("ERROR: Inline source map only works with singular input");
|
||||||
|
process.exit(1);
|
||||||
|
}
|
||||||
|
if (ARGS.acorn || ARGS.spidermonkey) {
|
||||||
|
print_error("ERROR: Inline source map only works with built-in parser");
|
||||||
|
process.exit(1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if (files.indexOf("-") >= 0 && ARGS.source_map) {
|
if (files.indexOf("-") >= 0 && ARGS.source_map) {
|
||||||
print_error("ERROR: Source map doesn't work with input from STDIN");
|
print_error("ERROR: Source map doesn't work with input from STDIN");
|
||||||
process.exit(1);
|
process.exit(1);
|
||||||
@@ -303,37 +316,19 @@ if (files.filter(function(el){ return el == "-" }).length > 1) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
var STATS = {};
|
var STATS = {};
|
||||||
var OUTPUT_FILE = ARGS.o;
|
|
||||||
var TOPLEVEL = null;
|
var TOPLEVEL = null;
|
||||||
var P_RELATIVE = ARGS.p && ARGS.p == "relative";
|
var P_RELATIVE = ARGS.p && ARGS.p == "relative";
|
||||||
var SOURCES_CONTENT = {};
|
var SOURCES_CONTENT = {};
|
||||||
|
|
||||||
var SOURCE_MAP = ARGS.source_map ? UglifyJS.SourceMap({
|
|
||||||
file: P_RELATIVE ? path.relative(path.dirname(ARGS.source_map), OUTPUT_FILE) : OUTPUT_FILE,
|
|
||||||
root: ARGS.source_map_root,
|
|
||||||
orig: ORIG_MAP,
|
|
||||||
}) : null;
|
|
||||||
|
|
||||||
OUTPUT_OPTIONS.source_map = SOURCE_MAP;
|
|
||||||
|
|
||||||
try {
|
|
||||||
var output = UglifyJS.OutputStream(OUTPUT_OPTIONS);
|
|
||||||
var compressor = COMPRESS && UglifyJS.Compressor(COMPRESS);
|
|
||||||
} catch(ex) {
|
|
||||||
if (ex instanceof UglifyJS.DefaultsError) {
|
|
||||||
print_error(ex.msg);
|
|
||||||
print_error("Supported options:");
|
|
||||||
print_error(sys.inspect(ex.defs));
|
|
||||||
process.exit(1);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
async.eachLimit(files, 1, function (file, cb) {
|
async.eachLimit(files, 1, function (file, cb) {
|
||||||
read_whole_file(file, function (err, code) {
|
read_whole_file(file, function (err, code) {
|
||||||
if (err) {
|
if (err) {
|
||||||
print_error("ERROR: can't read file: " + file);
|
print_error("ERROR: can't read file: " + file);
|
||||||
process.exit(1);
|
process.exit(1);
|
||||||
}
|
}
|
||||||
|
if (ORIG_MAP == "inline") {
|
||||||
|
ORIG_MAP = read_source_map(code);
|
||||||
|
}
|
||||||
if (ARGS.p != null) {
|
if (ARGS.p != null) {
|
||||||
if (P_RELATIVE) {
|
if (P_RELATIVE) {
|
||||||
file = path.relative(path.dirname(ARGS.source_map), file).replace(/\\/g, '/');
|
file = path.relative(path.dirname(ARGS.source_map), file).replace(/\\/g, '/');
|
||||||
@@ -369,7 +364,21 @@ async.eachLimit(files, 1, function (file, cb) {
|
|||||||
} catch(ex) {
|
} catch(ex) {
|
||||||
if (ex instanceof UglifyJS.JS_Parse_Error) {
|
if (ex instanceof UglifyJS.JS_Parse_Error) {
|
||||||
print_error("Parse error at " + file + ":" + ex.line + "," + ex.col);
|
print_error("Parse error at " + file + ":" + ex.line + "," + ex.col);
|
||||||
print_error(ex.message);
|
var col = ex.col;
|
||||||
|
var line = code.split(/\r?\n/)[ex.line - (col ? 1 : 2)];
|
||||||
|
if (line) {
|
||||||
|
if (col > 40) {
|
||||||
|
line = line.slice(col - 40);
|
||||||
|
col = 40;
|
||||||
|
}
|
||||||
|
if (col) {
|
||||||
|
print_error(line.slice(0, 80));
|
||||||
|
print_error(line.slice(0, col).replace(/\S/g, " ") + "^");
|
||||||
|
} else {
|
||||||
|
print_error(line.slice(-40));
|
||||||
|
print_error(line.slice(-40).replace(/\S/g, " ") + "^");
|
||||||
|
}
|
||||||
|
}
|
||||||
print_error(ex.stack);
|
print_error(ex.stack);
|
||||||
process.exit(1);
|
process.exit(1);
|
||||||
}
|
}
|
||||||
@@ -380,6 +389,28 @@ async.eachLimit(files, 1, function (file, cb) {
|
|||||||
cb();
|
cb();
|
||||||
});
|
});
|
||||||
}, function () {
|
}, function () {
|
||||||
|
var OUTPUT_FILE = ARGS.o;
|
||||||
|
|
||||||
|
var SOURCE_MAP = (ARGS.source_map || ARGS.source_map_inline) ? UglifyJS.SourceMap({
|
||||||
|
file: P_RELATIVE ? path.relative(path.dirname(ARGS.source_map), OUTPUT_FILE) : OUTPUT_FILE,
|
||||||
|
root: ARGS.source_map_root || ORIG_MAP && ORIG_MAP.sourceRoot,
|
||||||
|
orig: ORIG_MAP,
|
||||||
|
}) : null;
|
||||||
|
|
||||||
|
OUTPUT_OPTIONS.source_map = SOURCE_MAP;
|
||||||
|
|
||||||
|
try {
|
||||||
|
var output = UglifyJS.OutputStream(OUTPUT_OPTIONS);
|
||||||
|
var compressor = COMPRESS && UglifyJS.Compressor(COMPRESS);
|
||||||
|
} catch(ex) {
|
||||||
|
if (ex instanceof UglifyJS.DefaultsError) {
|
||||||
|
print_error(ex.message);
|
||||||
|
print_error("Supported options:");
|
||||||
|
print_error(sys.inspect(ex.defs));
|
||||||
|
process.exit(1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if (ARGS.acorn || ARGS.spidermonkey) time_it("convert_ast", function(){
|
if (ARGS.acorn || ARGS.spidermonkey) time_it("convert_ast", function(){
|
||||||
TOPLEVEL = UglifyJS.AST_Node.from_mozilla_ast(TOPLEVEL);
|
TOPLEVEL = UglifyJS.AST_Node.from_mozilla_ast(TOPLEVEL);
|
||||||
});
|
});
|
||||||
@@ -416,17 +447,19 @@ async.eachLimit(files, 1, function (file, cb) {
|
|||||||
cache : cache,
|
cache : cache,
|
||||||
only_cache : !ARGS.mangle_props,
|
only_cache : !ARGS.mangle_props,
|
||||||
regex : regex,
|
regex : regex,
|
||||||
ignore_quoted : ARGS.mangle_props == 2
|
ignore_quoted : ARGS.mangle_props == 2,
|
||||||
|
debug : typeof ARGS.mangle_props_debug === "undefined" ? false : ARGS.mangle_props_debug
|
||||||
});
|
});
|
||||||
writeNameCache("props", cache);
|
writeNameCache("props", cache);
|
||||||
})();
|
})();
|
||||||
|
|
||||||
var SCOPE_IS_NEEDED = COMPRESS || MANGLE || ARGS.lint
|
var SCOPE_IS_NEEDED = COMPRESS || MANGLE || ARGS.lint
|
||||||
var TL_CACHE = readNameCache("vars");
|
var TL_CACHE = readNameCache("vars");
|
||||||
|
if (MANGLE) MANGLE.cache = TL_CACHE;
|
||||||
|
|
||||||
if (SCOPE_IS_NEEDED) {
|
if (SCOPE_IS_NEEDED) {
|
||||||
time_it("scope", function(){
|
time_it("scope", function(){
|
||||||
TOPLEVEL.figure_out_scope({ screw_ie8: ARGS.screw_ie8, cache: TL_CACHE });
|
TOPLEVEL.figure_out_scope(MANGLE || { screw_ie8: screw_ie8, cache: TL_CACHE });
|
||||||
if (ARGS.lint) {
|
if (ARGS.lint) {
|
||||||
TOPLEVEL.scope_warnings();
|
TOPLEVEL.scope_warnings();
|
||||||
}
|
}
|
||||||
@@ -441,7 +474,7 @@ async.eachLimit(files, 1, function (file, cb) {
|
|||||||
|
|
||||||
if (SCOPE_IS_NEEDED) {
|
if (SCOPE_IS_NEEDED) {
|
||||||
time_it("scope", function(){
|
time_it("scope", function(){
|
||||||
TOPLEVEL.figure_out_scope({ screw_ie8: ARGS.screw_ie8, cache: TL_CACHE });
|
TOPLEVEL.figure_out_scope(MANGLE || { screw_ie8: screw_ie8, cache: TL_CACHE });
|
||||||
if (MANGLE && !TL_CACHE) {
|
if (MANGLE && !TL_CACHE) {
|
||||||
TOPLEVEL.compute_char_frequency(MANGLE);
|
TOPLEVEL.compute_char_frequency(MANGLE);
|
||||||
}
|
}
|
||||||
@@ -449,7 +482,6 @@ async.eachLimit(files, 1, function (file, cb) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (MANGLE) time_it("mangle", function(){
|
if (MANGLE) time_it("mangle", function(){
|
||||||
MANGLE.cache = TL_CACHE;
|
|
||||||
TOPLEVEL.mangle_names(MANGLE);
|
TOPLEVEL.mangle_names(MANGLE);
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -473,13 +505,18 @@ async.eachLimit(files, 1, function (file, cb) {
|
|||||||
output = output.get();
|
output = output.get();
|
||||||
|
|
||||||
if (SOURCE_MAP) {
|
if (SOURCE_MAP) {
|
||||||
fs.writeFileSync(ARGS.source_map, SOURCE_MAP, "utf8");
|
if (ARGS.source_map_inline) {
|
||||||
var source_map_url = ARGS.source_map_url || (
|
var base64_string = new Buffer(SOURCE_MAP.toString()).toString('base64');
|
||||||
P_RELATIVE
|
output += "\n//# sourceMappingURL=data:application/json;charset=utf-8;base64," + base64_string;
|
||||||
? path.relative(path.dirname(OUTPUT_FILE), ARGS.source_map)
|
} else {
|
||||||
: ARGS.source_map
|
fs.writeFileSync(ARGS.source_map, SOURCE_MAP, "utf8");
|
||||||
);
|
var source_map_url = ARGS.source_map_url || (
|
||||||
output += "\n//# sourceMappingURL=" + source_map_url;
|
P_RELATIVE
|
||||||
|
? path.relative(path.dirname(OUTPUT_FILE), ARGS.source_map)
|
||||||
|
: ARGS.source_map
|
||||||
|
);
|
||||||
|
output += "\n//# sourceMappingURL=" + source_map_url;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (OUTPUT_FILE) {
|
if (OUTPUT_FILE) {
|
||||||
@@ -530,7 +567,7 @@ function getOptions(flag, constants) {
|
|||||||
ast.walk(new UglifyJS.TreeWalker(function(node){
|
ast.walk(new UglifyJS.TreeWalker(function(node){
|
||||||
if (node instanceof UglifyJS.AST_Seq) return; // descend
|
if (node instanceof UglifyJS.AST_Seq) return; // descend
|
||||||
if (node instanceof UglifyJS.AST_Assign) {
|
if (node instanceof UglifyJS.AST_Assign) {
|
||||||
var name = node.left.print_to_string({ beautify: false }).replace(/-/g, "_");
|
var name = node.left.print_to_string().replace(/-/g, "_");
|
||||||
var value = node.right;
|
var value = node.right;
|
||||||
if (constants)
|
if (constants)
|
||||||
value = new Function("return (" + value.print_to_string() + ")")();
|
value = new Function("return (" + value.print_to_string() + ")")();
|
||||||
@@ -538,7 +575,7 @@ function getOptions(flag, constants) {
|
|||||||
return true; // no descend
|
return true; // no descend
|
||||||
}
|
}
|
||||||
if (node instanceof UglifyJS.AST_Symbol || node instanceof UglifyJS.AST_Binary) {
|
if (node instanceof UglifyJS.AST_Symbol || node instanceof UglifyJS.AST_Binary) {
|
||||||
var name = node.print_to_string({ beautify: false }).replace(/-/g, "_");
|
var name = node.print_to_string().replace(/-/g, "_");
|
||||||
ret[name] = true;
|
ret[name] = true;
|
||||||
return true; // no descend
|
return true; // no descend
|
||||||
}
|
}
|
||||||
@@ -565,6 +602,15 @@ function read_whole_file(filename, cb) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function read_source_map(code) {
|
||||||
|
var match = /\n\/\/# sourceMappingURL=data:application\/json(;.*?)?;base64,(.*)/.exec(code);
|
||||||
|
if (!match) {
|
||||||
|
print_error("WARN: inline source map not found");
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
return JSON.parse(new Buffer(match[2], "base64"));
|
||||||
|
}
|
||||||
|
|
||||||
function time_it(name, cont) {
|
function time_it(name, cont) {
|
||||||
var t1 = new Date().getTime();
|
var t1 = new Date().getTime();
|
||||||
var ret = cont();
|
var ret = cont();
|
||||||
|
|||||||
50
lib/ast.js
50
lib/ast.js
@@ -81,7 +81,9 @@ function DEFNODE(type, props, methods, base) {
|
|||||||
ctor.DEFMETHOD = function(name, method) {
|
ctor.DEFMETHOD = function(name, method) {
|
||||||
this.prototype[name] = method;
|
this.prototype[name] = method;
|
||||||
};
|
};
|
||||||
exports["AST_" + type] = ctor;
|
if (typeof exports !== "undefined") {
|
||||||
|
exports["AST_" + type] = ctor;
|
||||||
|
}
|
||||||
return ctor;
|
return ctor;
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -143,12 +145,13 @@ var AST_SimpleStatement = DEFNODE("SimpleStatement", "body", {
|
|||||||
}, AST_Statement);
|
}, AST_Statement);
|
||||||
|
|
||||||
function walk_body(node, visitor) {
|
function walk_body(node, visitor) {
|
||||||
if (node.body instanceof AST_Statement) {
|
var body = node.body;
|
||||||
node.body._walk(visitor);
|
if (body instanceof AST_Statement) {
|
||||||
|
body._walk(visitor);
|
||||||
|
}
|
||||||
|
else for (var i = 0, len = body.length; i < len; i++) {
|
||||||
|
body[i]._walk(visitor);
|
||||||
}
|
}
|
||||||
else node.body.forEach(function(stat){
|
|
||||||
stat._walk(visitor);
|
|
||||||
});
|
|
||||||
};
|
};
|
||||||
|
|
||||||
var AST_Block = DEFNODE("Block", "body", {
|
var AST_Block = DEFNODE("Block", "body", {
|
||||||
@@ -369,9 +372,10 @@ var AST_Lambda = DEFNODE("Lambda", "name argnames uses_arguments", {
|
|||||||
_walk: function(visitor) {
|
_walk: function(visitor) {
|
||||||
return visitor._visit(this, function(){
|
return visitor._visit(this, function(){
|
||||||
if (this.name) this.name._walk(visitor);
|
if (this.name) this.name._walk(visitor);
|
||||||
this.argnames.forEach(function(arg){
|
var argnames = this.argnames;
|
||||||
arg._walk(visitor);
|
for (var i = 0, len = argnames.length; i < len; i++) {
|
||||||
});
|
argnames[i]._walk(visitor);
|
||||||
|
}
|
||||||
walk_body(this, visitor);
|
walk_body(this, visitor);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
@@ -531,9 +535,10 @@ var AST_Definitions = DEFNODE("Definitions", "definitions", {
|
|||||||
},
|
},
|
||||||
_walk: function(visitor) {
|
_walk: function(visitor) {
|
||||||
return visitor._visit(this, function(){
|
return visitor._visit(this, function(){
|
||||||
this.definitions.forEach(function(def){
|
var definitions = this.definitions;
|
||||||
def._walk(visitor);
|
for (var i = 0, len = definitions.length; i < len; i++) {
|
||||||
});
|
definitions[i]._walk(visitor);
|
||||||
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
}, AST_Statement);
|
}, AST_Statement);
|
||||||
@@ -571,9 +576,10 @@ var AST_Call = DEFNODE("Call", "expression args", {
|
|||||||
_walk: function(visitor) {
|
_walk: function(visitor) {
|
||||||
return visitor._visit(this, function(){
|
return visitor._visit(this, function(){
|
||||||
this.expression._walk(visitor);
|
this.expression._walk(visitor);
|
||||||
this.args.forEach(function(arg){
|
var args = this.args;
|
||||||
arg._walk(visitor);
|
for (var i = 0, len = args.length; i < len; i++) {
|
||||||
});
|
args[i]._walk(visitor);
|
||||||
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
@@ -740,9 +746,10 @@ var AST_Array = DEFNODE("Array", "elements", {
|
|||||||
},
|
},
|
||||||
_walk: function(visitor) {
|
_walk: function(visitor) {
|
||||||
return visitor._visit(this, function(){
|
return visitor._visit(this, function(){
|
||||||
this.elements.forEach(function(el){
|
var elements = this.elements;
|
||||||
el._walk(visitor);
|
for (var i = 0, len = elements.length; i < len; i++) {
|
||||||
});
|
elements[i]._walk(visitor);
|
||||||
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
@@ -754,9 +761,10 @@ var AST_Object = DEFNODE("Object", "properties", {
|
|||||||
},
|
},
|
||||||
_walk: function(visitor) {
|
_walk: function(visitor) {
|
||||||
return visitor._visit(this, function(){
|
return visitor._visit(this, function(){
|
||||||
this.properties.forEach(function(prop){
|
var properties = this.properties;
|
||||||
prop._walk(visitor);
|
for (var i = 0, len = properties.length; i < len; i++) {
|
||||||
});
|
properties[i]._walk(visitor);
|
||||||
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|||||||
1006
lib/compress.js
1006
lib/compress.js
File diff suppressed because it is too large
Load Diff
@@ -45,20 +45,55 @@
|
|||||||
|
|
||||||
(function(){
|
(function(){
|
||||||
|
|
||||||
var MOZ_TO_ME = {
|
var normalize_directives = function(body) {
|
||||||
ExpressionStatement: function(M) {
|
var in_directive = true;
|
||||||
var expr = M.expression;
|
|
||||||
if (expr.type === "Literal" && typeof expr.value === "string") {
|
for (var i = 0; i < body.length; i++) {
|
||||||
return new AST_Directive({
|
if (in_directive && body[i] instanceof AST_Statement && body[i].body instanceof AST_String) {
|
||||||
start: my_start_token(M),
|
body[i] = new AST_Directive({
|
||||||
end: my_end_token(M),
|
start: body[i].start,
|
||||||
value: expr.value
|
end: body[i].end,
|
||||||
|
value: body[i].body.value
|
||||||
});
|
});
|
||||||
|
} else if (in_directive && !(body[i] instanceof AST_Statement && body[i].body instanceof AST_String)) {
|
||||||
|
in_directive = false;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return body;
|
||||||
|
};
|
||||||
|
|
||||||
|
var MOZ_TO_ME = {
|
||||||
|
Program: function(M) {
|
||||||
|
return new AST_Toplevel({
|
||||||
|
start: my_start_token(M),
|
||||||
|
end: my_end_token(M),
|
||||||
|
body: normalize_directives(M.body.map(from_moz))
|
||||||
|
});
|
||||||
|
},
|
||||||
|
FunctionDeclaration: function(M) {
|
||||||
|
return new AST_Defun({
|
||||||
|
start: my_start_token(M),
|
||||||
|
end: my_end_token(M),
|
||||||
|
name: from_moz(M.id),
|
||||||
|
argnames: M.params.map(from_moz),
|
||||||
|
body: normalize_directives(from_moz(M.body).body)
|
||||||
|
});
|
||||||
|
},
|
||||||
|
FunctionExpression: function(M) {
|
||||||
|
return new AST_Function({
|
||||||
|
start: my_start_token(M),
|
||||||
|
end: my_end_token(M),
|
||||||
|
name: from_moz(M.id),
|
||||||
|
argnames: M.params.map(from_moz),
|
||||||
|
body: normalize_directives(from_moz(M.body).body)
|
||||||
|
});
|
||||||
|
},
|
||||||
|
ExpressionStatement: function(M) {
|
||||||
return new AST_SimpleStatement({
|
return new AST_SimpleStatement({
|
||||||
start: my_start_token(M),
|
start: my_start_token(M),
|
||||||
end: my_end_token(M),
|
end: my_end_token(M),
|
||||||
body: from_moz(expr)
|
body: from_moz(M.expression)
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
TryStatement: function(M) {
|
TryStatement: function(M) {
|
||||||
@@ -94,6 +129,15 @@
|
|||||||
return new AST_ObjectGetter(args);
|
return new AST_ObjectGetter(args);
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
ArrayExpression: function(M) {
|
||||||
|
return new AST_Array({
|
||||||
|
start : my_start_token(M),
|
||||||
|
end : my_end_token(M),
|
||||||
|
elements : M.elements.map(function(elem){
|
||||||
|
return elem === null ? new AST_Hole() : from_moz(elem);
|
||||||
|
})
|
||||||
|
});
|
||||||
|
},
|
||||||
ObjectExpression: function(M) {
|
ObjectExpression: function(M) {
|
||||||
return new AST_Object({
|
return new AST_Object({
|
||||||
start : my_start_token(M),
|
start : my_start_token(M),
|
||||||
@@ -185,7 +229,6 @@
|
|||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
map("Program", AST_Toplevel, "body@body");
|
|
||||||
map("EmptyStatement", AST_EmptyStatement);
|
map("EmptyStatement", AST_EmptyStatement);
|
||||||
map("BlockStatement", AST_BlockStatement, "body@body");
|
map("BlockStatement", AST_BlockStatement, "body@body");
|
||||||
map("IfStatement", AST_If, "test>condition, consequent>body, alternate>alternative");
|
map("IfStatement", AST_If, "test>condition, consequent>body, alternate>alternative");
|
||||||
@@ -201,13 +244,10 @@
|
|||||||
map("ForStatement", AST_For, "init>init, test>condition, update>step, body>body");
|
map("ForStatement", AST_For, "init>init, test>condition, update>step, body>body");
|
||||||
map("ForInStatement", AST_ForIn, "left>init, right>object, body>body");
|
map("ForInStatement", AST_ForIn, "left>init, right>object, body>body");
|
||||||
map("DebuggerStatement", AST_Debugger);
|
map("DebuggerStatement", AST_Debugger);
|
||||||
map("FunctionDeclaration", AST_Defun, "id>name, params@argnames, body%body");
|
|
||||||
map("VariableDeclarator", AST_VarDef, "id>name, init>value");
|
map("VariableDeclarator", AST_VarDef, "id>name, init>value");
|
||||||
map("CatchClause", AST_Catch, "param>argname, body%body");
|
map("CatchClause", AST_Catch, "param>argname, body%body");
|
||||||
|
|
||||||
map("ThisExpression", AST_This);
|
map("ThisExpression", AST_This);
|
||||||
map("ArrayExpression", AST_Array, "elements@elements");
|
|
||||||
map("FunctionExpression", AST_Function, "id>name, params@argnames, body%body");
|
|
||||||
map("BinaryExpression", AST_Binary, "operator=operator, left>left, right>right");
|
map("BinaryExpression", AST_Binary, "operator=operator, left>left, right>right");
|
||||||
map("LogicalExpression", AST_Binary, "operator=operator, left>left, right>right");
|
map("LogicalExpression", AST_Binary, "operator=operator, left>left, right>right");
|
||||||
map("AssignmentExpression", AST_Assign, "operator=operator, left>left, right>right");
|
map("AssignmentExpression", AST_Assign, "operator=operator, left>left, right>right");
|
||||||
@@ -215,6 +255,31 @@
|
|||||||
map("NewExpression", AST_New, "callee>expression, arguments@args");
|
map("NewExpression", AST_New, "callee>expression, arguments@args");
|
||||||
map("CallExpression", AST_Call, "callee>expression, arguments@args");
|
map("CallExpression", AST_Call, "callee>expression, arguments@args");
|
||||||
|
|
||||||
|
def_to_moz(AST_Toplevel, function To_Moz_Program(M) {
|
||||||
|
return {
|
||||||
|
type: "Program",
|
||||||
|
body: M.body.map(to_moz)
|
||||||
|
};
|
||||||
|
});
|
||||||
|
|
||||||
|
def_to_moz(AST_Defun, function To_Moz_FunctionDeclaration(M) {
|
||||||
|
return {
|
||||||
|
type: "FunctionDeclaration",
|
||||||
|
id: to_moz(M.name),
|
||||||
|
params: M.argnames.map(to_moz),
|
||||||
|
body: to_moz_block(M)
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
def_to_moz(AST_Function, function To_Moz_FunctionExpression(M) {
|
||||||
|
return {
|
||||||
|
type: "FunctionExpression",
|
||||||
|
id: to_moz(M.name),
|
||||||
|
params: M.argnames.map(to_moz),
|
||||||
|
body: to_moz_block(M)
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
def_to_moz(AST_Directive, function To_Moz_Directive(M) {
|
def_to_moz(AST_Directive, function To_Moz_Directive(M) {
|
||||||
return {
|
return {
|
||||||
type: "ExpressionStatement",
|
type: "ExpressionStatement",
|
||||||
@@ -302,6 +367,13 @@
|
|||||||
};
|
};
|
||||||
});
|
});
|
||||||
|
|
||||||
|
def_to_moz(AST_Array, function To_Moz_ArrayExpression(M) {
|
||||||
|
return {
|
||||||
|
type: "ArrayExpression",
|
||||||
|
elements: M.elements.map(to_moz)
|
||||||
|
};
|
||||||
|
});
|
||||||
|
|
||||||
def_to_moz(AST_Object, function To_Moz_ObjectExpression(M) {
|
def_to_moz(AST_Object, function To_Moz_ObjectExpression(M) {
|
||||||
return {
|
return {
|
||||||
type: "ObjectExpression",
|
type: "ObjectExpression",
|
||||||
|
|||||||
159
lib/output.js
159
lib/output.js
@@ -45,6 +45,20 @@
|
|||||||
|
|
||||||
var EXPECT_DIRECTIVE = /^$|[;{][\s\n]*$/;
|
var EXPECT_DIRECTIVE = /^$|[;{][\s\n]*$/;
|
||||||
|
|
||||||
|
function is_some_comments(comment) {
|
||||||
|
var text = comment.value;
|
||||||
|
var type = comment.type;
|
||||||
|
if (type == "comment2") {
|
||||||
|
// multiline comment
|
||||||
|
return /@preserve|@license|@cc_on/i.test(text);
|
||||||
|
}
|
||||||
|
return type == "comment5";
|
||||||
|
}
|
||||||
|
|
||||||
|
function is_comment5(comment) {
|
||||||
|
return comment.type == "comment5";
|
||||||
|
}
|
||||||
|
|
||||||
function OutputStream(options) {
|
function OutputStream(options) {
|
||||||
|
|
||||||
options = defaults(options, {
|
options = defaults(options, {
|
||||||
@@ -56,7 +70,7 @@ function OutputStream(options) {
|
|||||||
unescape_regexps : false,
|
unescape_regexps : false,
|
||||||
inline_script : false,
|
inline_script : false,
|
||||||
width : 80,
|
width : 80,
|
||||||
max_line_len : 32000,
|
max_line_len : false,
|
||||||
beautify : false,
|
beautify : false,
|
||||||
source_map : null,
|
source_map : null,
|
||||||
bracketize : false,
|
bracketize : false,
|
||||||
@@ -64,12 +78,41 @@ function OutputStream(options) {
|
|||||||
comments : false,
|
comments : false,
|
||||||
shebang : true,
|
shebang : true,
|
||||||
preserve_line : false,
|
preserve_line : false,
|
||||||
screw_ie8 : false,
|
screw_ie8 : true,
|
||||||
preamble : null,
|
preamble : null,
|
||||||
quote_style : 0,
|
quote_style : 0,
|
||||||
keep_quoted_props: false
|
keep_quoted_props: false,
|
||||||
|
wrap_iife : false,
|
||||||
}, true);
|
}, true);
|
||||||
|
|
||||||
|
// Convert comment option to RegExp if neccessary and set up comments filter
|
||||||
|
var comment_filter = options.shebang ? is_comment5 : return_false; // Default case, throw all comments away except shebangs
|
||||||
|
if (options.comments) {
|
||||||
|
var comments = options.comments;
|
||||||
|
if (typeof options.comments === "string" && /^\/.*\/[a-zA-Z]*$/.test(options.comments)) {
|
||||||
|
var regex_pos = options.comments.lastIndexOf("/");
|
||||||
|
comments = new RegExp(
|
||||||
|
options.comments.substr(1, regex_pos - 1),
|
||||||
|
options.comments.substr(regex_pos + 1)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
if (comments instanceof RegExp) {
|
||||||
|
comment_filter = function(comment) {
|
||||||
|
return comment.type == "comment5" || comments.test(comment.value);
|
||||||
|
};
|
||||||
|
}
|
||||||
|
else if (typeof comments === "function") {
|
||||||
|
comment_filter = function(comment) {
|
||||||
|
return comment.type == "comment5" || comments(this, comment);
|
||||||
|
};
|
||||||
|
}
|
||||||
|
else if (comments === "some") {
|
||||||
|
comment_filter = is_some_comments;
|
||||||
|
} else { // NOTE includes "all" option
|
||||||
|
comment_filter = return_true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
var indentation = 0;
|
var indentation = 0;
|
||||||
var current_col = 0;
|
var current_col = 0;
|
||||||
var current_line = 1;
|
var current_line = 1;
|
||||||
@@ -155,16 +198,29 @@ function OutputStream(options) {
|
|||||||
|
|
||||||
var might_need_space = false;
|
var might_need_space = false;
|
||||||
var might_need_semicolon = false;
|
var might_need_semicolon = false;
|
||||||
|
var might_add_newline = 0;
|
||||||
var last = null;
|
var last = null;
|
||||||
|
|
||||||
function last_char() {
|
function last_char() {
|
||||||
return last.charAt(last.length - 1);
|
return last.charAt(last.length - 1);
|
||||||
};
|
};
|
||||||
|
|
||||||
function maybe_newline() {
|
var ensure_line_len = options.max_line_len ? function() {
|
||||||
if (options.max_line_len && current_col > options.max_line_len)
|
if (current_col > options.max_line_len) {
|
||||||
print("\n");
|
if (might_add_newline) {
|
||||||
};
|
var left = OUTPUT.slice(0, might_add_newline);
|
||||||
|
var right = OUTPUT.slice(might_add_newline);
|
||||||
|
OUTPUT = left + "\n" + right;
|
||||||
|
current_line++;
|
||||||
|
current_pos++;
|
||||||
|
current_col = right.length;
|
||||||
|
}
|
||||||
|
if (current_col > options.max_line_len) {
|
||||||
|
AST_Node.warn("Output exceeds {max_line_len} characters", options);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
might_add_newline = 0;
|
||||||
|
} : noop;
|
||||||
|
|
||||||
var requireSemicolonChars = makePredicate("( [ + * / - , .");
|
var requireSemicolonChars = makePredicate("( [ + * / - , .");
|
||||||
|
|
||||||
@@ -180,6 +236,7 @@ function OutputStream(options) {
|
|||||||
current_col++;
|
current_col++;
|
||||||
current_pos++;
|
current_pos++;
|
||||||
} else {
|
} else {
|
||||||
|
ensure_line_len();
|
||||||
OUTPUT += "\n";
|
OUTPUT += "\n";
|
||||||
current_pos++;
|
current_pos++;
|
||||||
current_line++;
|
current_line++;
|
||||||
@@ -200,6 +257,7 @@ function OutputStream(options) {
|
|||||||
if (!options.beautify && options.preserve_line && stack[stack.length - 1]) {
|
if (!options.beautify && options.preserve_line && stack[stack.length - 1]) {
|
||||||
var target_line = stack[stack.length - 1].start.line;
|
var target_line = stack[stack.length - 1].start.line;
|
||||||
while (current_line < target_line) {
|
while (current_line < target_line) {
|
||||||
|
ensure_line_len();
|
||||||
OUTPUT += "\n";
|
OUTPUT += "\n";
|
||||||
current_pos++;
|
current_pos++;
|
||||||
current_line++;
|
current_line++;
|
||||||
@@ -211,8 +269,9 @@ function OutputStream(options) {
|
|||||||
if (might_need_space) {
|
if (might_need_space) {
|
||||||
var prev = last_char();
|
var prev = last_char();
|
||||||
if ((is_identifier_char(prev)
|
if ((is_identifier_char(prev)
|
||||||
&& (is_identifier_char(ch) || ch == "\\"))
|
&& (is_identifier_char(ch) || ch == "\\"))
|
||||||
|| (/^[\+\-\/]$/.test(ch) && ch == prev))
|
|| (ch == "/" && ch == prev)
|
||||||
|
|| ((ch == "+" || ch == "-") && ch == last))
|
||||||
{
|
{
|
||||||
OUTPUT += " ";
|
OUTPUT += " ";
|
||||||
current_col++;
|
current_col++;
|
||||||
@@ -220,16 +279,16 @@ function OutputStream(options) {
|
|||||||
}
|
}
|
||||||
might_need_space = false;
|
might_need_space = false;
|
||||||
}
|
}
|
||||||
|
OUTPUT += str;
|
||||||
|
current_pos += str.length;
|
||||||
var a = str.split(/\r?\n/), n = a.length - 1;
|
var a = str.split(/\r?\n/), n = a.length - 1;
|
||||||
current_line += n;
|
current_line += n;
|
||||||
if (n == 0) {
|
current_col += a[0].length;
|
||||||
current_col += a[n].length;
|
if (n > 0) {
|
||||||
} else {
|
ensure_line_len();
|
||||||
current_col = a[n].length;
|
current_col = a[n].length;
|
||||||
}
|
}
|
||||||
current_pos += str.length;
|
|
||||||
last = str;
|
last = str;
|
||||||
OUTPUT += str;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
var space = options.beautify ? function() {
|
var space = options.beautify ? function() {
|
||||||
@@ -255,7 +314,10 @@ function OutputStream(options) {
|
|||||||
|
|
||||||
var newline = options.beautify ? function() {
|
var newline = options.beautify ? function() {
|
||||||
print("\n");
|
print("\n");
|
||||||
} : maybe_newline;
|
} : options.max_line_len ? function() {
|
||||||
|
ensure_line_len();
|
||||||
|
might_add_newline = OUTPUT.length;
|
||||||
|
} : noop;
|
||||||
|
|
||||||
var semicolon = options.beautify ? function() {
|
var semicolon = options.beautify ? function() {
|
||||||
print(";");
|
print(";");
|
||||||
@@ -332,6 +394,9 @@ function OutputStream(options) {
|
|||||||
} : noop;
|
} : noop;
|
||||||
|
|
||||||
function get() {
|
function get() {
|
||||||
|
if (might_add_newline) {
|
||||||
|
ensure_line_len();
|
||||||
|
}
|
||||||
return OUTPUT;
|
return OUTPUT;
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -376,12 +441,12 @@ function OutputStream(options) {
|
|||||||
with_square : with_square,
|
with_square : with_square,
|
||||||
add_mapping : add_mapping,
|
add_mapping : add_mapping,
|
||||||
option : function(opt) { return options[opt] },
|
option : function(opt) { return options[opt] },
|
||||||
|
comment_filter : comment_filter,
|
||||||
line : function() { return current_line },
|
line : function() { return current_line },
|
||||||
col : function() { return current_col },
|
col : function() { return current_col },
|
||||||
pos : function() { return current_pos },
|
pos : function() { return current_pos },
|
||||||
push_node : function(node) { stack.push(node) },
|
push_node : function(node) { stack.push(node) },
|
||||||
pop_node : function() { return stack.pop() },
|
pop_node : function() { return stack.pop() },
|
||||||
stack : function() { return stack },
|
|
||||||
parent : function(n) {
|
parent : function(n) {
|
||||||
return stack[stack.length - 2 - (n || 0)];
|
return stack[stack.length - 2 - (n || 0)];
|
||||||
}
|
}
|
||||||
@@ -435,7 +500,7 @@ function OutputStream(options) {
|
|||||||
|
|
||||||
AST_Node.DEFMETHOD("add_comments", function(output){
|
AST_Node.DEFMETHOD("add_comments", function(output){
|
||||||
if (output._readonly) return;
|
if (output._readonly) return;
|
||||||
var c = output.option("comments"), self = this;
|
var self = this;
|
||||||
var start = self.start;
|
var start = self.start;
|
||||||
if (start && !start._comments_dumped) {
|
if (start && !start._comments_dumped) {
|
||||||
start._comments_dumped = true;
|
start._comments_dumped = true;
|
||||||
@@ -458,19 +523,7 @@ function OutputStream(options) {
|
|||||||
}));
|
}));
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!c) {
|
comments = comments.filter(output.comment_filter, self);
|
||||||
comments = comments.filter(function(comment) {
|
|
||||||
return comment.type == "comment5";
|
|
||||||
});
|
|
||||||
} else if (c.test) {
|
|
||||||
comments = comments.filter(function(comment){
|
|
||||||
return comment.type == "comment5" || c.test(comment.value);
|
|
||||||
});
|
|
||||||
} else if (typeof c == "function") {
|
|
||||||
comments = comments.filter(function(comment){
|
|
||||||
return comment.type == "comment5" || c(self, comment);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
// Keep single line comments after nlb, after nlb
|
// Keep single line comments after nlb, after nlb
|
||||||
if (!output.option("beautify") && comments.length > 0 &&
|
if (!output.option("beautify") && comments.length > 0 &&
|
||||||
@@ -521,7 +574,16 @@ function OutputStream(options) {
|
|||||||
// a function expression needs parens around it when it's provably
|
// a function expression needs parens around it when it's provably
|
||||||
// the first token to appear in a statement.
|
// the first token to appear in a statement.
|
||||||
PARENS(AST_Function, function(output){
|
PARENS(AST_Function, function(output){
|
||||||
return first_in_statement(output);
|
if (first_in_statement(output)) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (output.option('wrap_iife')) {
|
||||||
|
var p = output.parent();
|
||||||
|
return p instanceof AST_Call && p.expression === this;
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
});
|
});
|
||||||
|
|
||||||
// same goes for an object literal, because otherwise it would be
|
// same goes for an object literal, because otherwise it would be
|
||||||
@@ -533,7 +595,7 @@ function OutputStream(options) {
|
|||||||
PARENS([ AST_Unary, AST_Undefined ], function(output){
|
PARENS([ AST_Unary, AST_Undefined ], function(output){
|
||||||
var p = output.parent();
|
var p = output.parent();
|
||||||
return p instanceof AST_PropAccess && p.expression === this
|
return p instanceof AST_PropAccess && p.expression === this
|
||||||
|| p instanceof AST_New;
|
|| p instanceof AST_Call && p.expression === this;
|
||||||
});
|
});
|
||||||
|
|
||||||
PARENS(AST_Seq, function(output){
|
PARENS(AST_Seq, function(output){
|
||||||
@@ -861,8 +923,8 @@ function OutputStream(options) {
|
|||||||
// adds the block brackets if needed.
|
// adds the block brackets if needed.
|
||||||
if (!self.body)
|
if (!self.body)
|
||||||
return output.force_semicolon();
|
return output.force_semicolon();
|
||||||
if (self.body instanceof AST_Do
|
if (self.body instanceof AST_Do) {
|
||||||
&& !output.option("screw_ie8")) {
|
// Unconditionally use the if/do-while workaround for all browsers.
|
||||||
// https://github.com/mishoo/UglifyJS/issues/#issue/57 IE
|
// https://github.com/mishoo/UglifyJS/issues/#issue/57 IE
|
||||||
// croaks with "syntax error" on code like this: if (foo)
|
// croaks with "syntax error" on code like this: if (foo)
|
||||||
// do ... while(cond); else ... we need block brackets
|
// do ... while(cond); else ... we need block brackets
|
||||||
@@ -898,7 +960,10 @@ function OutputStream(options) {
|
|||||||
output.space();
|
output.space();
|
||||||
output.print("else");
|
output.print("else");
|
||||||
output.space();
|
output.space();
|
||||||
force_statement(self.alternative, output);
|
if (self.alternative instanceof AST_If)
|
||||||
|
self.alternative.print(output);
|
||||||
|
else
|
||||||
|
force_statement(self.alternative, output);
|
||||||
} else {
|
} else {
|
||||||
self._do_print_body(output);
|
self._do_print_body(output);
|
||||||
}
|
}
|
||||||
@@ -1293,30 +1358,6 @@ function OutputStream(options) {
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
// return true if the node at the top of the stack (that means the
|
|
||||||
// innermost node in the current output) is lexically the first in
|
|
||||||
// a statement.
|
|
||||||
function first_in_statement(output) {
|
|
||||||
var a = output.stack(), i = a.length, node = a[--i], p = a[--i];
|
|
||||||
while (i > 0) {
|
|
||||||
if (p instanceof AST_Statement && p.body === node)
|
|
||||||
return true;
|
|
||||||
if ((p instanceof AST_Seq && p.car === node ) ||
|
|
||||||
(p instanceof AST_Call && p.expression === node && !(p instanceof AST_New) ) ||
|
|
||||||
(p instanceof AST_Dot && p.expression === node ) ||
|
|
||||||
(p instanceof AST_Sub && p.expression === node ) ||
|
|
||||||
(p instanceof AST_Conditional && p.condition === node ) ||
|
|
||||||
(p instanceof AST_Binary && p.left === node ) ||
|
|
||||||
(p instanceof AST_UnaryPostfix && p.expression === node ))
|
|
||||||
{
|
|
||||||
node = p;
|
|
||||||
p = a[--i];
|
|
||||||
} else {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
// self should be AST_New. decide if we want to show parens or not.
|
// self should be AST_New. decide if we want to show parens or not.
|
||||||
function need_constructor_parens(self, output) {
|
function need_constructor_parens(self, output) {
|
||||||
// Always print parentheses with arguments
|
// Always print parentheses with arguments
|
||||||
|
|||||||
121
lib/parse.js
121
lib/parse.js
@@ -107,7 +107,9 @@ var OPERATORS = makePredicate([
|
|||||||
"||"
|
"||"
|
||||||
]);
|
]);
|
||||||
|
|
||||||
var WHITESPACE_CHARS = makePredicate(characters(" \u00a0\n\r\t\f\u000b\u200b\u180e\u2000\u2001\u2002\u2003\u2004\u2005\u2006\u2007\u2008\u2009\u200a\u202f\u205f\u3000\uFEFF"));
|
var WHITESPACE_CHARS = makePredicate(characters(" \u00a0\n\r\t\f\u000b\u200b\u2000\u2001\u2002\u2003\u2004\u2005\u2006\u2007\u2008\u2009\u200a\u2028\u2029\u202f\u205f\u3000\uFEFF"));
|
||||||
|
|
||||||
|
var NEWLINE_CHARS = makePredicate(characters("\n\r\u2028\u2029"));
|
||||||
|
|
||||||
var PUNC_BEFORE_EXPRESSION = makePredicate(characters("[{(,.;:"));
|
var PUNC_BEFORE_EXPRESSION = makePredicate(characters("[{(,.;:"));
|
||||||
|
|
||||||
@@ -193,12 +195,11 @@ function JS_Parse_Error(message, filename, line, col, pos) {
|
|||||||
this.line = line;
|
this.line = line;
|
||||||
this.col = col;
|
this.col = col;
|
||||||
this.pos = pos;
|
this.pos = pos;
|
||||||
this.stack = new Error().stack;
|
|
||||||
};
|
|
||||||
|
|
||||||
JS_Parse_Error.prototype.toString = function() {
|
|
||||||
return this.message + " (line: " + this.line + ", col: " + this.col + ", pos: " + this.pos + ")" + "\n\n" + this.stack;
|
|
||||||
};
|
};
|
||||||
|
JS_Parse_Error.prototype = Object.create(Error.prototype);
|
||||||
|
JS_Parse_Error.prototype.constructor = JS_Parse_Error;
|
||||||
|
JS_Parse_Error.prototype.name = "SyntaxError";
|
||||||
|
configure_error_stack(JS_Parse_Error);
|
||||||
|
|
||||||
function js_error(message, filename, line, col, pos) {
|
function js_error(message, filename, line, col, pos) {
|
||||||
throw new JS_Parse_Error(message, filename, line, col, pos);
|
throw new JS_Parse_Error(message, filename, line, col, pos);
|
||||||
@@ -234,7 +235,7 @@ function tokenizer($TEXT, filename, html5_comments, shebang) {
|
|||||||
var ch = S.text.charAt(S.pos++);
|
var ch = S.text.charAt(S.pos++);
|
||||||
if (signal_eof && !ch)
|
if (signal_eof && !ch)
|
||||||
throw EX_EOF;
|
throw EX_EOF;
|
||||||
if ("\r\n\u2028\u2029".indexOf(ch) >= 0) {
|
if (NEWLINE_CHARS(ch)) {
|
||||||
S.newline_before = S.newline_before || !in_string;
|
S.newline_before = S.newline_before || !in_string;
|
||||||
++S.line;
|
++S.line;
|
||||||
S.col = 0;
|
S.col = 0;
|
||||||
@@ -261,7 +262,7 @@ function tokenizer($TEXT, filename, html5_comments, shebang) {
|
|||||||
var text = S.text;
|
var text = S.text;
|
||||||
for (var i = S.pos, n = S.text.length; i < n; ++i) {
|
for (var i = S.pos, n = S.text.length; i < n; ++i) {
|
||||||
var ch = text[i];
|
var ch = text[i];
|
||||||
if (ch == '\n' || ch == '\r')
|
if (NEWLINE_CHARS(ch))
|
||||||
return i;
|
return i;
|
||||||
}
|
}
|
||||||
return -1;
|
return -1;
|
||||||
@@ -313,8 +314,7 @@ function tokenizer($TEXT, filename, html5_comments, shebang) {
|
|||||||
};
|
};
|
||||||
|
|
||||||
function skip_whitespace() {
|
function skip_whitespace() {
|
||||||
var ch;
|
while (WHITESPACE_CHARS(peek()))
|
||||||
while (WHITESPACE_CHARS(ch = peek()) || ch == "\u2028" || ch == "\u2029")
|
|
||||||
next();
|
next();
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -348,6 +348,9 @@ function tokenizer($TEXT, filename, html5_comments, shebang) {
|
|||||||
return is_alphanumeric_char(code);
|
return is_alphanumeric_char(code);
|
||||||
});
|
});
|
||||||
if (prefix) num = prefix + num;
|
if (prefix) num = prefix + num;
|
||||||
|
if (RE_OCT_NUMBER.test(num) && next_token.has_directive("use strict")) {
|
||||||
|
parse_error("Legacy octal literals are not allowed in strict mode");
|
||||||
|
}
|
||||||
var valid = parse_js_number(num);
|
var valid = parse_js_number(num);
|
||||||
if (!isNaN(valid)) {
|
if (!isNaN(valid)) {
|
||||||
return token("num", valid);
|
return token("num", valid);
|
||||||
@@ -365,7 +368,6 @@ function tokenizer($TEXT, filename, html5_comments, shebang) {
|
|||||||
case 98 : return "\b";
|
case 98 : return "\b";
|
||||||
case 118 : return "\u000b"; // \v
|
case 118 : return "\u000b"; // \v
|
||||||
case 102 : return "\f";
|
case 102 : return "\f";
|
||||||
case 48 : return "\0";
|
|
||||||
case 120 : return String.fromCharCode(hex_bytes(2)); // \x
|
case 120 : return String.fromCharCode(hex_bytes(2)); // \x
|
||||||
case 117 : return String.fromCharCode(hex_bytes(4)); // \u
|
case 117 : return String.fromCharCode(hex_bytes(4)); // \u
|
||||||
case 10 : return ""; // newline
|
case 10 : return ""; // newline
|
||||||
@@ -375,9 +377,27 @@ function tokenizer($TEXT, filename, html5_comments, shebang) {
|
|||||||
return "";
|
return "";
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
if (ch >= "0" && ch <= "7")
|
||||||
|
return read_octal_escape_sequence(ch);
|
||||||
return ch;
|
return ch;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
function read_octal_escape_sequence(ch) {
|
||||||
|
// Read
|
||||||
|
var p = peek();
|
||||||
|
if (p >= "0" && p <= "7") {
|
||||||
|
ch += next(true);
|
||||||
|
if (ch[0] <= "3" && (p = peek()) >= "0" && p <= "7")
|
||||||
|
ch += next(true);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Parse
|
||||||
|
if (ch === "0") return "\0";
|
||||||
|
if (ch.length > 0 && next_token.has_directive("use strict"))
|
||||||
|
parse_error("Legacy octal escape sequences are not allowed in strict mode");
|
||||||
|
return String.fromCharCode(parseInt(ch, 8));
|
||||||
|
}
|
||||||
|
|
||||||
function hex_bytes(n) {
|
function hex_bytes(n) {
|
||||||
var num = 0;
|
var num = 0;
|
||||||
for (; n > 0; --n) {
|
for (; n > 0; --n) {
|
||||||
@@ -393,28 +413,8 @@ function tokenizer($TEXT, filename, html5_comments, shebang) {
|
|||||||
var quote = next(), ret = "";
|
var quote = next(), ret = "";
|
||||||
for (;;) {
|
for (;;) {
|
||||||
var ch = next(true, true);
|
var ch = next(true, true);
|
||||||
if (ch == "\\") {
|
if (ch == "\\") ch = read_escaped_char(true);
|
||||||
var octal_len = 0, first = null;
|
else if (NEWLINE_CHARS(ch)) parse_error("Unterminated string constant");
|
||||||
ch = read_while(function(ch){
|
|
||||||
if (ch >= "0" && ch <= "7") {
|
|
||||||
if (!first) {
|
|
||||||
first = ch;
|
|
||||||
return ++octal_len;
|
|
||||||
}
|
|
||||||
else if (first <= "3" && octal_len <= 2) return ++octal_len;
|
|
||||||
else if (first >= "4" && octal_len <= 1) return ++octal_len;
|
|
||||||
}
|
|
||||||
return false;
|
|
||||||
});
|
|
||||||
if (octal_len > 0) {
|
|
||||||
if (ch !== "0" && next_token.has_directive("use strict"))
|
|
||||||
parse_error("Octal literals are not allowed in strict mode");
|
|
||||||
ch = String.fromCharCode(parseInt(ch, 8));
|
|
||||||
} else {
|
|
||||||
ch = read_escaped_char(true);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else if ("\r\n\u2028\u2029".indexOf(ch) >= 0) parse_error("Unterminated string constant");
|
|
||||||
else if (ch == quote) break;
|
else if (ch == quote) break;
|
||||||
ret += ch;
|
ret += ch;
|
||||||
}
|
}
|
||||||
@@ -442,18 +442,11 @@ function tokenizer($TEXT, filename, html5_comments, shebang) {
|
|||||||
var skip_multiline_comment = with_eof_error("Unterminated multiline comment", function(){
|
var skip_multiline_comment = with_eof_error("Unterminated multiline comment", function(){
|
||||||
var regex_allowed = S.regex_allowed;
|
var regex_allowed = S.regex_allowed;
|
||||||
var i = find("*/", true);
|
var i = find("*/", true);
|
||||||
var text = S.text.substring(S.pos, i).replace(/\r\n|\r/g, '\n');
|
var text = S.text.substring(S.pos, i).replace(/\r\n|\r|\u2028|\u2029/g, '\n');
|
||||||
var a = text.split("\n"), n = a.length;
|
|
||||||
// update stream position
|
// update stream position
|
||||||
S.pos = i + 2;
|
forward(text.length /* doesn't count \r\n as 2 char while S.pos - i does */ + 2);
|
||||||
S.line += n - 1;
|
|
||||||
if (n > 1) S.col = a[n - 1].length;
|
|
||||||
else S.col += a[n - 1].length;
|
|
||||||
S.col += 2;
|
|
||||||
var nlb = S.newline_before = S.newline_before || text.indexOf("\n") >= 0;
|
|
||||||
S.comments_before.push(token("comment2", text, true));
|
S.comments_before.push(token("comment2", text, true));
|
||||||
S.regex_allowed = regex_allowed;
|
S.regex_allowed = regex_allowed;
|
||||||
S.newline_before = nlb;
|
|
||||||
return next_token;
|
return next_token;
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -482,7 +475,9 @@ function tokenizer($TEXT, filename, html5_comments, shebang) {
|
|||||||
|
|
||||||
var read_regexp = with_eof_error("Unterminated regular expression", function(regexp){
|
var read_regexp = with_eof_error("Unterminated regular expression", function(regexp){
|
||||||
var prev_backslash = false, ch, in_class = false;
|
var prev_backslash = false, ch, in_class = false;
|
||||||
while ((ch = next(true))) if (prev_backslash) {
|
while ((ch = next(true))) if (NEWLINE_CHARS(ch)) {
|
||||||
|
parse_error("Unexpected line terminator");
|
||||||
|
} else if (prev_backslash) {
|
||||||
regexp += "\\" + ch;
|
regexp += "\\" + ch;
|
||||||
prev_backslash = false;
|
prev_backslash = false;
|
||||||
} else if (ch == "[") {
|
} else if (ch == "[") {
|
||||||
@@ -495,8 +490,6 @@ function tokenizer($TEXT, filename, html5_comments, shebang) {
|
|||||||
break;
|
break;
|
||||||
} else if (ch == "\\") {
|
} else if (ch == "\\") {
|
||||||
prev_backslash = true;
|
prev_backslash = true;
|
||||||
} else if ("\r\n\u2028\u2029".indexOf(ch) >= 0) {
|
|
||||||
parse_error("Unexpected line terminator");
|
|
||||||
} else {
|
} else {
|
||||||
regexp += ch;
|
regexp += ch;
|
||||||
}
|
}
|
||||||
@@ -762,14 +755,14 @@ function parse($TEXT, options) {
|
|||||||
function unexpected(token) {
|
function unexpected(token) {
|
||||||
if (token == null)
|
if (token == null)
|
||||||
token = S.token;
|
token = S.token;
|
||||||
token_error(token, "SyntaxError: Unexpected token: " + token.type + " (" + token.value + ")");
|
token_error(token, "Unexpected token: " + token.type + " (" + token.value + ")");
|
||||||
};
|
};
|
||||||
|
|
||||||
function expect_token(type, val) {
|
function expect_token(type, val) {
|
||||||
if (is(type, val)) {
|
if (is(type, val)) {
|
||||||
return next();
|
return next();
|
||||||
}
|
}
|
||||||
token_error(S.token, "SyntaxError: Unexpected token " + S.token.type + " «" + S.token.value + "»" + ", expected " + type + " «" + val + "»");
|
token_error(S.token, "Unexpected token " + S.token.type + " «" + S.token.value + "»" + ", expected " + type + " «" + val + "»");
|
||||||
};
|
};
|
||||||
|
|
||||||
function expect(punc) { return expect_token("punc", punc); };
|
function expect(punc) { return expect_token("punc", punc); };
|
||||||
@@ -898,7 +891,7 @@ function parse($TEXT, options) {
|
|||||||
|
|
||||||
case "return":
|
case "return":
|
||||||
if (S.in_function == 0 && !options.bare_returns)
|
if (S.in_function == 0 && !options.bare_returns)
|
||||||
croak("SyntaxError: 'return' outside of function");
|
croak("'return' outside of function");
|
||||||
return new AST_Return({
|
return new AST_Return({
|
||||||
value: ( is("punc", ";")
|
value: ( is("punc", ";")
|
||||||
? (next(), null)
|
? (next(), null)
|
||||||
@@ -915,7 +908,7 @@ function parse($TEXT, options) {
|
|||||||
|
|
||||||
case "throw":
|
case "throw":
|
||||||
if (S.token.nlb)
|
if (S.token.nlb)
|
||||||
croak("SyntaxError: Illegal newline after 'throw'");
|
croak("Illegal newline after 'throw'");
|
||||||
return new AST_Throw({
|
return new AST_Throw({
|
||||||
value: (tmp = expression(true), semicolon(), tmp)
|
value: (tmp = expression(true), semicolon(), tmp)
|
||||||
});
|
});
|
||||||
@@ -931,7 +924,7 @@ function parse($TEXT, options) {
|
|||||||
|
|
||||||
case "with":
|
case "with":
|
||||||
if (S.input.has_directive("use strict")) {
|
if (S.input.has_directive("use strict")) {
|
||||||
croak("SyntaxError: Strict mode may not include a with statement");
|
croak("Strict mode may not include a with statement");
|
||||||
}
|
}
|
||||||
return new AST_With({
|
return new AST_With({
|
||||||
expression : parenthesised(),
|
expression : parenthesised(),
|
||||||
@@ -951,7 +944,7 @@ function parse($TEXT, options) {
|
|||||||
// syntactically incorrect if it contains a
|
// syntactically incorrect if it contains a
|
||||||
// LabelledStatement that is enclosed by a
|
// LabelledStatement that is enclosed by a
|
||||||
// LabelledStatement with the same Identifier as label.
|
// LabelledStatement with the same Identifier as label.
|
||||||
croak("SyntaxError: Label " + label.name + " defined twice");
|
croak("Label " + label.name + " defined twice");
|
||||||
}
|
}
|
||||||
expect(":");
|
expect(":");
|
||||||
S.labels.push(label);
|
S.labels.push(label);
|
||||||
@@ -964,7 +957,7 @@ function parse($TEXT, options) {
|
|||||||
label.references.forEach(function(ref){
|
label.references.forEach(function(ref){
|
||||||
if (ref instanceof AST_Continue) {
|
if (ref instanceof AST_Continue) {
|
||||||
ref = ref.label.start;
|
ref = ref.label.start;
|
||||||
croak("SyntaxError: Continue label `" + label.name + "` refers to non-IterationStatement.",
|
croak("Continue label `" + label.name + "` refers to non-IterationStatement.",
|
||||||
ref.line, ref.col, ref.pos);
|
ref.line, ref.col, ref.pos);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
@@ -984,11 +977,11 @@ function parse($TEXT, options) {
|
|||||||
if (label != null) {
|
if (label != null) {
|
||||||
ldef = find_if(function(l){ return l.name == label.name }, S.labels);
|
ldef = find_if(function(l){ return l.name == label.name }, S.labels);
|
||||||
if (!ldef)
|
if (!ldef)
|
||||||
croak("SyntaxError: Undefined label " + label.name);
|
croak("Undefined label " + label.name);
|
||||||
label.thedef = ldef;
|
label.thedef = ldef;
|
||||||
}
|
}
|
||||||
else if (S.in_loop == 0)
|
else if (S.in_loop == 0)
|
||||||
croak("SyntaxError: " + type.TYPE + " not inside a loop or switch");
|
croak(type.TYPE + " not inside a loop or switch");
|
||||||
semicolon();
|
semicolon();
|
||||||
var stat = new type({ label: label });
|
var stat = new type({ label: label });
|
||||||
if (ldef) ldef.references.push(stat);
|
if (ldef) ldef.references.push(stat);
|
||||||
@@ -1004,7 +997,7 @@ function parse($TEXT, options) {
|
|||||||
: expression(true, true);
|
: expression(true, true);
|
||||||
if (is("operator", "in")) {
|
if (is("operator", "in")) {
|
||||||
if (init instanceof AST_Var && init.definitions.length > 1)
|
if (init instanceof AST_Var && init.definitions.length > 1)
|
||||||
croak("SyntaxError: Only one variable declaration allowed in for..in loop");
|
croak("Only one variable declaration allowed in for..in loop");
|
||||||
next();
|
next();
|
||||||
return for_in(init);
|
return for_in(init);
|
||||||
}
|
}
|
||||||
@@ -1154,7 +1147,7 @@ function parse($TEXT, options) {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
if (!bcatch && !bfinally)
|
if (!bcatch && !bfinally)
|
||||||
croak("SyntaxError: Missing catch/finally blocks");
|
croak("Missing catch/finally blocks");
|
||||||
return new AST_Try({
|
return new AST_Try({
|
||||||
body : body,
|
body : body,
|
||||||
bcatch : bcatch,
|
bcatch : bcatch,
|
||||||
@@ -1248,7 +1241,7 @@ function parse($TEXT, options) {
|
|||||||
break;
|
break;
|
||||||
case "operator":
|
case "operator":
|
||||||
if (!is_identifier_string(tok.value)) {
|
if (!is_identifier_string(tok.value)) {
|
||||||
croak("SyntaxError: Invalid getter/setter name: " + tok.value,
|
croak("Invalid getter/setter name: " + tok.value,
|
||||||
tok.line, tok.col, tok.pos);
|
tok.line, tok.col, tok.pos);
|
||||||
}
|
}
|
||||||
ret = _make_symbol(AST_SymbolRef);
|
ret = _make_symbol(AST_SymbolRef);
|
||||||
@@ -1314,6 +1307,10 @@ function parse($TEXT, options) {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
var create_accessor = embed_tokens(function() {
|
||||||
|
return function_(AST_Accessor);
|
||||||
|
});
|
||||||
|
|
||||||
var object_ = embed_tokens(function() {
|
var object_ = embed_tokens(function() {
|
||||||
expect("{");
|
expect("{");
|
||||||
var first = true, a = [];
|
var first = true, a = [];
|
||||||
@@ -1330,7 +1327,7 @@ function parse($TEXT, options) {
|
|||||||
a.push(new AST_ObjectGetter({
|
a.push(new AST_ObjectGetter({
|
||||||
start : start,
|
start : start,
|
||||||
key : as_atom_node(),
|
key : as_atom_node(),
|
||||||
value : function_(AST_Accessor),
|
value : create_accessor(),
|
||||||
end : prev()
|
end : prev()
|
||||||
}));
|
}));
|
||||||
continue;
|
continue;
|
||||||
@@ -1339,7 +1336,7 @@ function parse($TEXT, options) {
|
|||||||
a.push(new AST_ObjectSetter({
|
a.push(new AST_ObjectSetter({
|
||||||
start : start,
|
start : start,
|
||||||
key : as_atom_node(),
|
key : as_atom_node(),
|
||||||
value : function_(AST_Accessor),
|
value : create_accessor(),
|
||||||
end : prev()
|
end : prev()
|
||||||
}));
|
}));
|
||||||
continue;
|
continue;
|
||||||
@@ -1399,7 +1396,7 @@ function parse($TEXT, options) {
|
|||||||
|
|
||||||
function as_symbol(type, noerror) {
|
function as_symbol(type, noerror) {
|
||||||
if (!is("name")) {
|
if (!is("name")) {
|
||||||
if (!noerror) croak("SyntaxError: Name expected");
|
if (!noerror) croak("Name expected");
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
var sym = _make_symbol(type);
|
var sym = _make_symbol(type);
|
||||||
@@ -1463,7 +1460,7 @@ function parse($TEXT, options) {
|
|||||||
|
|
||||||
function make_unary(ctor, op, expr) {
|
function make_unary(ctor, op, expr) {
|
||||||
if ((op == "++" || op == "--") && !is_assignable(expr))
|
if ((op == "++" || op == "--") && !is_assignable(expr))
|
||||||
croak("SyntaxError: Invalid use of " + op + " operator");
|
croak("Invalid use of " + op + " operator");
|
||||||
return new ctor({ operator: op, expression: expr });
|
return new ctor({ operator: op, expression: expr });
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -1527,7 +1524,7 @@ function parse($TEXT, options) {
|
|||||||
end : prev()
|
end : prev()
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
croak("SyntaxError: Invalid assignment");
|
croak("Invalid assignment");
|
||||||
}
|
}
|
||||||
return left;
|
return left;
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -66,7 +66,8 @@ function mangle_properties(ast, options) {
|
|||||||
cache : null,
|
cache : null,
|
||||||
only_cache : false,
|
only_cache : false,
|
||||||
regex : null,
|
regex : null,
|
||||||
ignore_quoted : false
|
ignore_quoted : false,
|
||||||
|
debug : false
|
||||||
});
|
});
|
||||||
|
|
||||||
var reserved = options.reserved;
|
var reserved = options.reserved;
|
||||||
@@ -84,29 +85,33 @@ function mangle_properties(ast, options) {
|
|||||||
var regex = options.regex;
|
var regex = options.regex;
|
||||||
var ignore_quoted = options.ignore_quoted;
|
var ignore_quoted = options.ignore_quoted;
|
||||||
|
|
||||||
|
// note debug is either false (disabled), or a string of the debug suffix to use (enabled).
|
||||||
|
// note debug may be enabled as an empty string, which is falsey. Also treat passing 'true'
|
||||||
|
// the same as passing an empty string.
|
||||||
|
var debug = (options.debug !== false);
|
||||||
|
var debug_name_suffix;
|
||||||
|
if (debug) {
|
||||||
|
debug_name_suffix = (options.debug === true ? "" : options.debug);
|
||||||
|
}
|
||||||
|
|
||||||
var names_to_mangle = [];
|
var names_to_mangle = [];
|
||||||
var unmangleable = [];
|
var unmangleable = [];
|
||||||
|
var ignored = {};
|
||||||
|
|
||||||
// step 1: find candidates to mangle
|
// step 1: find candidates to mangle
|
||||||
ast.walk(new TreeWalker(function(node){
|
ast.walk(new TreeWalker(function(node){
|
||||||
if (node instanceof AST_ObjectKeyVal) {
|
if (node instanceof AST_ObjectKeyVal) {
|
||||||
if (!(ignore_quoted && node.quote))
|
add(node.key, ignore_quoted && node.quote);
|
||||||
add(node.key);
|
|
||||||
}
|
}
|
||||||
else if (node instanceof AST_ObjectProperty) {
|
else if (node instanceof AST_ObjectProperty) {
|
||||||
// setter or getter, since KeyVal is handled above
|
// setter or getter, since KeyVal is handled above
|
||||||
add(node.key.name);
|
add(node.key.name);
|
||||||
}
|
}
|
||||||
else if (node instanceof AST_Dot) {
|
else if (node instanceof AST_Dot) {
|
||||||
if (this.parent() instanceof AST_Assign) {
|
add(node.property);
|
||||||
add(node.property);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
else if (node instanceof AST_Sub) {
|
else if (node instanceof AST_Sub) {
|
||||||
if (this.parent() instanceof AST_Assign) {
|
addStrings(node.property, ignore_quoted);
|
||||||
if (!ignore_quoted)
|
|
||||||
addStrings(node.property);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}));
|
}));
|
||||||
|
|
||||||
@@ -144,6 +149,7 @@ function mangle_properties(ast, options) {
|
|||||||
// only function declarations after this line
|
// only function declarations after this line
|
||||||
|
|
||||||
function can_mangle(name) {
|
function can_mangle(name) {
|
||||||
|
if (!is_identifier(name)) return false;
|
||||||
if (unmangleable.indexOf(name) >= 0) return false;
|
if (unmangleable.indexOf(name) >= 0) return false;
|
||||||
if (reserved.indexOf(name) >= 0) return false;
|
if (reserved.indexOf(name) >= 0) return false;
|
||||||
if (options.only_cache) {
|
if (options.only_cache) {
|
||||||
@@ -154,13 +160,19 @@ function mangle_properties(ast, options) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
function should_mangle(name) {
|
function should_mangle(name) {
|
||||||
|
if (ignore_quoted && name in ignored) return false;
|
||||||
if (regex && !regex.test(name)) return false;
|
if (regex && !regex.test(name)) return false;
|
||||||
if (reserved.indexOf(name) >= 0) return false;
|
if (reserved.indexOf(name) >= 0) return false;
|
||||||
return cache.props.has(name)
|
return cache.props.has(name)
|
||||||
|| names_to_mangle.indexOf(name) >= 0;
|
|| names_to_mangle.indexOf(name) >= 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
function add(name) {
|
function add(name, ignore) {
|
||||||
|
if (ignore) {
|
||||||
|
ignored[name] = true;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
if (can_mangle(name))
|
if (can_mangle(name))
|
||||||
push_uniq(names_to_mangle, name);
|
push_uniq(names_to_mangle, name);
|
||||||
|
|
||||||
@@ -176,15 +188,31 @@ function mangle_properties(ast, options) {
|
|||||||
|
|
||||||
var mangled = cache.props.get(name);
|
var mangled = cache.props.get(name);
|
||||||
if (!mangled) {
|
if (!mangled) {
|
||||||
do {
|
if (debug) {
|
||||||
mangled = base54(++cache.cname);
|
// debug mode: use a prefix and suffix to preserve readability, e.g. o.foo -> o._$foo$NNN_.
|
||||||
} while (!can_mangle(mangled));
|
var debug_mangled = "_$" + name + "$" + debug_name_suffix + "_";
|
||||||
|
|
||||||
|
if (can_mangle(debug_mangled) && !(ignore_quoted && debug_mangled in ignored)) {
|
||||||
|
mangled = debug_mangled;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// either debug mode is off, or it is on and we could not use the mangled name
|
||||||
|
if (!mangled) {
|
||||||
|
// note can_mangle() does not check if the name collides with the 'ignored' set
|
||||||
|
// (filled with quoted properties when ignore_quoted set). Make sure we add this
|
||||||
|
// check so we don't collide with a quoted name.
|
||||||
|
do {
|
||||||
|
mangled = base54(++cache.cname);
|
||||||
|
} while (!can_mangle(mangled) || (ignore_quoted && mangled in ignored));
|
||||||
|
}
|
||||||
|
|
||||||
cache.props.set(name, mangled);
|
cache.props.set(name, mangled);
|
||||||
}
|
}
|
||||||
return mangled;
|
return mangled;
|
||||||
}
|
}
|
||||||
|
|
||||||
function addStrings(node) {
|
function addStrings(node, ignore) {
|
||||||
var out = {};
|
var out = {};
|
||||||
try {
|
try {
|
||||||
(function walk(node){
|
(function walk(node){
|
||||||
@@ -194,7 +222,7 @@ function mangle_properties(ast, options) {
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
if (node instanceof AST_String) {
|
if (node instanceof AST_String) {
|
||||||
add(node.value);
|
add(node.value, ignore);
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
if (node instanceof AST_Conditional) {
|
if (node instanceof AST_Conditional) {
|
||||||
|
|||||||
82
lib/scope.js
82
lib/scope.js
@@ -51,7 +51,6 @@ function SymbolDef(scope, index, orig) {
|
|||||||
this.global = false;
|
this.global = false;
|
||||||
this.mangled_name = null;
|
this.mangled_name = null;
|
||||||
this.undeclared = false;
|
this.undeclared = false;
|
||||||
this.constant = false;
|
|
||||||
this.index = index;
|
this.index = index;
|
||||||
this.id = SymbolDef.next_id++;
|
this.id = SymbolDef.next_id++;
|
||||||
};
|
};
|
||||||
@@ -88,7 +87,7 @@ SymbolDef.prototype = {
|
|||||||
|
|
||||||
AST_Toplevel.DEFMETHOD("figure_out_scope", function(options){
|
AST_Toplevel.DEFMETHOD("figure_out_scope", function(options){
|
||||||
options = defaults(options, {
|
options = defaults(options, {
|
||||||
screw_ie8: false,
|
screw_ie8: true,
|
||||||
cache: null
|
cache: null
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -97,26 +96,24 @@ AST_Toplevel.DEFMETHOD("figure_out_scope", function(options){
|
|||||||
var scope = self.parent_scope = null;
|
var scope = self.parent_scope = null;
|
||||||
var labels = new Dictionary();
|
var labels = new Dictionary();
|
||||||
var defun = null;
|
var defun = null;
|
||||||
var last_var_had_const_pragma = false;
|
|
||||||
var nesting = 0;
|
|
||||||
var tw = new TreeWalker(function(node, descend){
|
var tw = new TreeWalker(function(node, descend){
|
||||||
if (options.screw_ie8 && node instanceof AST_Catch) {
|
if (options.screw_ie8 && node instanceof AST_Catch) {
|
||||||
var save_scope = scope;
|
var save_scope = scope;
|
||||||
scope = new AST_Scope(node);
|
scope = new AST_Scope(node);
|
||||||
scope.init_scope_vars(nesting);
|
scope.init_scope_vars();
|
||||||
scope.parent_scope = save_scope;
|
scope.parent_scope = save_scope;
|
||||||
descend();
|
descend();
|
||||||
scope = save_scope;
|
scope = save_scope;
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
if (node instanceof AST_Scope) {
|
if (node instanceof AST_Scope) {
|
||||||
node.init_scope_vars(nesting);
|
node.init_scope_vars();
|
||||||
var save_scope = node.parent_scope = scope;
|
var save_scope = node.parent_scope = scope;
|
||||||
var save_defun = defun;
|
var save_defun = defun;
|
||||||
var save_labels = labels;
|
var save_labels = labels;
|
||||||
defun = scope = node;
|
defun = scope = node;
|
||||||
labels = new Dictionary();
|
labels = new Dictionary();
|
||||||
++nesting; descend(); --nesting;
|
descend();
|
||||||
scope = save_scope;
|
scope = save_scope;
|
||||||
defun = save_defun;
|
defun = save_defun;
|
||||||
labels = save_labels;
|
labels = save_labels;
|
||||||
@@ -155,13 +152,9 @@ AST_Toplevel.DEFMETHOD("figure_out_scope", function(options){
|
|||||||
// later.
|
// later.
|
||||||
(node.scope = defun.parent_scope).def_function(node);
|
(node.scope = defun.parent_scope).def_function(node);
|
||||||
}
|
}
|
||||||
else if (node instanceof AST_Var) {
|
|
||||||
last_var_had_const_pragma = node.has_const_pragma();
|
|
||||||
}
|
|
||||||
else if (node instanceof AST_SymbolVar
|
else if (node instanceof AST_SymbolVar
|
||||||
|| node instanceof AST_SymbolConst) {
|
|| node instanceof AST_SymbolConst) {
|
||||||
var def = defun.def_variable(node);
|
var def = defun.def_variable(node);
|
||||||
def.constant = node instanceof AST_SymbolConst || last_var_had_const_pragma;
|
|
||||||
def.init = tw.parent().value;
|
def.init = tw.parent().value;
|
||||||
}
|
}
|
||||||
else if (node instanceof AST_SymbolCatch) {
|
else if (node instanceof AST_SymbolCatch) {
|
||||||
@@ -203,24 +196,14 @@ AST_Toplevel.DEFMETHOD("figure_out_scope", function(options){
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
var sym = node.scope.find_variable(name);
|
var sym = node.scope.find_variable(name);
|
||||||
if (!sym) {
|
if (node.scope instanceof AST_Lambda && name == "arguments") {
|
||||||
var g;
|
node.scope.uses_arguments = true;
|
||||||
if (globals.has(name)) {
|
|
||||||
g = globals.get(name);
|
|
||||||
} else {
|
|
||||||
g = new SymbolDef(self, globals.size(), node);
|
|
||||||
g.undeclared = true;
|
|
||||||
g.global = true;
|
|
||||||
globals.set(name, g);
|
|
||||||
}
|
|
||||||
node.thedef = g;
|
|
||||||
if (func && name == "arguments") {
|
|
||||||
func.uses_arguments = true;
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
node.thedef = sym;
|
|
||||||
}
|
}
|
||||||
node.reference();
|
if (!sym) {
|
||||||
|
sym = self.def_global(node);
|
||||||
|
}
|
||||||
|
node.thedef = sym;
|
||||||
|
node.reference(options);
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
@@ -231,7 +214,20 @@ AST_Toplevel.DEFMETHOD("figure_out_scope", function(options){
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
AST_Scope.DEFMETHOD("init_scope_vars", function(nesting){
|
AST_Toplevel.DEFMETHOD("def_global", function(node){
|
||||||
|
var globals = this.globals, name = node.name;
|
||||||
|
if (globals.has(name)) {
|
||||||
|
return globals.get(name);
|
||||||
|
} else {
|
||||||
|
var g = new SymbolDef(this, globals.size(), node);
|
||||||
|
g.undeclared = true;
|
||||||
|
g.global = true;
|
||||||
|
globals.set(name, g);
|
||||||
|
return g;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
AST_Scope.DEFMETHOD("init_scope_vars", function(){
|
||||||
this.variables = new Dictionary(); // map name to AST_SymbolVar (variables defined in this scope; includes functions)
|
this.variables = new Dictionary(); // map name to AST_SymbolVar (variables defined in this scope; includes functions)
|
||||||
this.functions = new Dictionary(); // map name to AST_SymbolDefun (functions defined in this scope)
|
this.functions = new Dictionary(); // map name to AST_SymbolDefun (functions defined in this scope)
|
||||||
this.uses_with = false; // will be set to true if this or some nested scope uses the `with` statement
|
this.uses_with = false; // will be set to true if this or some nested scope uses the `with` statement
|
||||||
@@ -239,7 +235,6 @@ AST_Scope.DEFMETHOD("init_scope_vars", function(nesting){
|
|||||||
this.parent_scope = null; // the parent scope
|
this.parent_scope = null; // the parent scope
|
||||||
this.enclosed = []; // a list of variables from this or outer scope(s) that are referenced from this or inner scopes
|
this.enclosed = []; // a list of variables from this or outer scope(s) that are referenced from this or inner scopes
|
||||||
this.cname = -1; // the current index for mangling functions/variables
|
this.cname = -1; // the current index for mangling functions/variables
|
||||||
this.nesting = nesting; // the nesting level of this scope (0 means toplevel)
|
|
||||||
});
|
});
|
||||||
|
|
||||||
AST_Lambda.DEFMETHOD("init_scope_vars", function(){
|
AST_Lambda.DEFMETHOD("init_scope_vars", function(){
|
||||||
@@ -251,16 +246,20 @@ AST_Lambda.DEFMETHOD("init_scope_vars", function(){
|
|||||||
this.variables.set(symbol.name, def);
|
this.variables.set(symbol.name, def);
|
||||||
});
|
});
|
||||||
|
|
||||||
AST_SymbolRef.DEFMETHOD("reference", function() {
|
AST_SymbolRef.DEFMETHOD("reference", function(options) {
|
||||||
var def = this.definition();
|
var def = this.definition();
|
||||||
def.references.push(this);
|
def.references.push(this);
|
||||||
var s = this.scope;
|
var s = this.scope;
|
||||||
while (s) {
|
while (s) {
|
||||||
push_uniq(s.enclosed, def);
|
push_uniq(s.enclosed, def);
|
||||||
|
if (options.keep_fnames) {
|
||||||
|
s.functions.each(function(d) {
|
||||||
|
push_uniq(def.scope.enclosed, d);
|
||||||
|
});
|
||||||
|
}
|
||||||
if (s === def.scope) break;
|
if (s === def.scope) break;
|
||||||
s = s.parent_scope;
|
s = s.parent_scope;
|
||||||
}
|
}
|
||||||
this.frame = this.scope.nesting - def.scope.nesting;
|
|
||||||
});
|
});
|
||||||
|
|
||||||
AST_Scope.DEFMETHOD("find_variable", function(name){
|
AST_Scope.DEFMETHOD("find_variable", function(name){
|
||||||
@@ -314,18 +313,17 @@ AST_Function.DEFMETHOD("next_mangled", function(options, def){
|
|||||||
// a function expression's argument cannot shadow the function expression's name
|
// a function expression's argument cannot shadow the function expression's name
|
||||||
|
|
||||||
var tricky_def = def.orig[0] instanceof AST_SymbolFunarg && this.name && this.name.definition();
|
var tricky_def = def.orig[0] instanceof AST_SymbolFunarg && this.name && this.name.definition();
|
||||||
|
|
||||||
|
// the function's mangled_name is null when keep_fnames is true
|
||||||
|
var tricky_name = tricky_def ? tricky_def.mangled_name || tricky_def.name : null;
|
||||||
|
|
||||||
while (true) {
|
while (true) {
|
||||||
var name = AST_Lambda.prototype.next_mangled.call(this, options, def);
|
var name = AST_Lambda.prototype.next_mangled.call(this, options, def);
|
||||||
if (!(tricky_def && tricky_def.mangled_name == name))
|
if (!tricky_name || tricky_name != name)
|
||||||
return name;
|
return name;
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
AST_Scope.DEFMETHOD("references", function(sym){
|
|
||||||
if (sym instanceof AST_Symbol) sym = sym.definition();
|
|
||||||
return this.enclosed.indexOf(sym) < 0 ? null : sym;
|
|
||||||
});
|
|
||||||
|
|
||||||
AST_Symbol.DEFMETHOD("unmangleable", function(options){
|
AST_Symbol.DEFMETHOD("unmangleable", function(options){
|
||||||
return this.definition().unmangleable(options);
|
return this.definition().unmangleable(options);
|
||||||
});
|
});
|
||||||
@@ -365,19 +363,13 @@ AST_Symbol.DEFMETHOD("global", function(){
|
|||||||
return this.definition().global;
|
return this.definition().global;
|
||||||
});
|
});
|
||||||
|
|
||||||
AST_Var.DEFMETHOD("has_const_pragma", function() {
|
|
||||||
var comments_before = this.start && this.start.comments_before;
|
|
||||||
var lastComment = comments_before && comments_before[comments_before.length - 1];
|
|
||||||
return lastComment && /@const\b/.test(lastComment.value);
|
|
||||||
});
|
|
||||||
|
|
||||||
AST_Toplevel.DEFMETHOD("_default_mangler_options", function(options){
|
AST_Toplevel.DEFMETHOD("_default_mangler_options", function(options){
|
||||||
return defaults(options, {
|
return defaults(options, {
|
||||||
except : [],
|
except : [],
|
||||||
eval : false,
|
eval : false,
|
||||||
sort : false, // Ignored. Flag retained for backwards compatibility.
|
sort : false, // Ignored. Flag retained for backwards compatibility.
|
||||||
toplevel : false,
|
toplevel : false,
|
||||||
screw_ie8 : false,
|
screw_ie8 : true,
|
||||||
keep_fnames : false
|
keep_fnames : false
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -58,6 +58,16 @@ function SourceMap(options) {
|
|||||||
sourceRoot : options.root
|
sourceRoot : options.root
|
||||||
});
|
});
|
||||||
var orig_map = options.orig && new MOZ_SourceMap.SourceMapConsumer(options.orig);
|
var orig_map = options.orig && new MOZ_SourceMap.SourceMapConsumer(options.orig);
|
||||||
|
|
||||||
|
if (orig_map && Array.isArray(options.orig.sources)) {
|
||||||
|
orig_map._sources.toArray().forEach(function(source) {
|
||||||
|
var sourceContent = orig_map.sourceContentFor(source, true);
|
||||||
|
if (sourceContent) {
|
||||||
|
generator.setSourceContent(source, sourceContent);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
function add(source, gen_line, gen_col, orig_line, orig_col, name) {
|
function add(source, gen_line, gen_col, orig_line, orig_col, name) {
|
||||||
if (orig_map) {
|
if (orig_map) {
|
||||||
var info = orig_map.originalPositionFor({
|
var info = orig_map.originalPositionFor({
|
||||||
|
|||||||
57
lib/utils.js
57
lib/utils.js
@@ -78,13 +78,28 @@ function repeat_string(str, i) {
|
|||||||
return d;
|
return d;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
function configure_error_stack(fn) {
|
||||||
|
Object.defineProperty(fn.prototype, "stack", {
|
||||||
|
get: function() {
|
||||||
|
var err = new Error(this.message);
|
||||||
|
err.name = this.name;
|
||||||
|
try {
|
||||||
|
throw err;
|
||||||
|
} catch(e) {
|
||||||
|
return e.stack;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
function DefaultsError(msg, defs) {
|
function DefaultsError(msg, defs) {
|
||||||
Error.call(this, msg);
|
this.message = msg;
|
||||||
this.msg = msg;
|
|
||||||
this.defs = defs;
|
this.defs = defs;
|
||||||
};
|
};
|
||||||
DefaultsError.prototype = Object.create(Error.prototype);
|
DefaultsError.prototype = Object.create(Error.prototype);
|
||||||
DefaultsError.prototype.constructor = DefaultsError;
|
DefaultsError.prototype.constructor = DefaultsError;
|
||||||
|
DefaultsError.prototype.name = "DefaultsError";
|
||||||
|
configure_error_stack(DefaultsError);
|
||||||
|
|
||||||
DefaultsError.croak = function(msg, defs) {
|
DefaultsError.croak = function(msg, defs) {
|
||||||
throw new DefaultsError(msg, defs);
|
throw new DefaultsError(msg, defs);
|
||||||
@@ -112,6 +127,8 @@ function merge(obj, ext) {
|
|||||||
};
|
};
|
||||||
|
|
||||||
function noop() {};
|
function noop() {};
|
||||||
|
function return_false() { return false; }
|
||||||
|
function return_true() { return true; }
|
||||||
|
|
||||||
var MAP = (function(){
|
var MAP = (function(){
|
||||||
function MAP(a, f, backwards) {
|
function MAP(a, f, backwards) {
|
||||||
@@ -227,10 +244,19 @@ function makePredicate(words) {
|
|||||||
}
|
}
|
||||||
cats.push([words[i]]);
|
cats.push([words[i]]);
|
||||||
}
|
}
|
||||||
|
function quote(word) {
|
||||||
|
return JSON.stringify(word).replace(/[\u2028\u2029]/g, function(s) {
|
||||||
|
switch (s) {
|
||||||
|
case "\u2028": return "\\u2028";
|
||||||
|
case "\u2029": return "\\u2029";
|
||||||
|
}
|
||||||
|
return s;
|
||||||
|
});
|
||||||
|
}
|
||||||
function compareTo(arr) {
|
function compareTo(arr) {
|
||||||
if (arr.length == 1) return f += "return str === " + JSON.stringify(arr[0]) + ";";
|
if (arr.length == 1) return f += "return str === " + quote(arr[0]) + ";";
|
||||||
f += "switch(str){";
|
f += "switch(str){";
|
||||||
for (var i = 0; i < arr.length; ++i) f += "case " + JSON.stringify(arr[i]) + ":";
|
for (var i = 0; i < arr.length; ++i) f += "case " + quote(arr[i]) + ":";
|
||||||
f += "return true}return false;";
|
f += "return true}return false;";
|
||||||
}
|
}
|
||||||
// When there are more than three length categories, an outer
|
// When there are more than three length categories, an outer
|
||||||
@@ -309,3 +335,26 @@ Dictionary.fromObject = function(obj) {
|
|||||||
function HOP(obj, prop) {
|
function HOP(obj, prop) {
|
||||||
return Object.prototype.hasOwnProperty.call(obj, prop);
|
return Object.prototype.hasOwnProperty.call(obj, prop);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// return true if the node at the top of the stack (that means the
|
||||||
|
// innermost node in the current output) is lexically the first in
|
||||||
|
// a statement.
|
||||||
|
function first_in_statement(stack) {
|
||||||
|
var node = stack.parent(-1);
|
||||||
|
for (var i = 0, p; p = stack.parent(i); i++) {
|
||||||
|
if (p instanceof AST_Statement && p.body === node)
|
||||||
|
return true;
|
||||||
|
if ((p instanceof AST_Seq && p.car === node ) ||
|
||||||
|
(p instanceof AST_Call && p.expression === node && !(p instanceof AST_New) ) ||
|
||||||
|
(p instanceof AST_Dot && p.expression === node ) ||
|
||||||
|
(p instanceof AST_Sub && p.expression === node ) ||
|
||||||
|
(p instanceof AST_Conditional && p.condition === node ) ||
|
||||||
|
(p instanceof AST_Binary && p.left === node ) ||
|
||||||
|
(p instanceof AST_UnaryPostfix && p.expression === node ))
|
||||||
|
{
|
||||||
|
node = p;
|
||||||
|
} else {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|||||||
128
npm-shrinkwrap.json
generated
128
npm-shrinkwrap.json
generated
@@ -1,128 +0,0 @@
|
|||||||
{
|
|
||||||
"name": "uglify-js",
|
|
||||||
"version": "2.4.24",
|
|
||||||
"dependencies": {
|
|
||||||
"abbrev": {
|
|
||||||
"version": "1.0.7",
|
|
||||||
"from": "abbrev@>=1.0.0 <2.0.0",
|
|
||||||
"resolved": "https://registry.npmjs.org/abbrev/-/abbrev-1.0.7.tgz"
|
|
||||||
},
|
|
||||||
"amdefine": {
|
|
||||||
"version": "1.0.0",
|
|
||||||
"from": "amdefine@>=0.0.4",
|
|
||||||
"resolved": "https://registry.npmjs.org/amdefine/-/amdefine-1.0.0.tgz"
|
|
||||||
},
|
|
||||||
"async": {
|
|
||||||
"version": "0.2.10",
|
|
||||||
"from": "async@>=0.2.6 <0.3.0",
|
|
||||||
"resolved": "https://registry.npmjs.org/async/-/async-0.2.10.tgz"
|
|
||||||
},
|
|
||||||
"camelcase": {
|
|
||||||
"version": "1.2.1",
|
|
||||||
"from": "camelcase@>=1.0.2 <2.0.0",
|
|
||||||
"resolved": "https://registry.npmjs.org/camelcase/-/camelcase-1.2.1.tgz"
|
|
||||||
},
|
|
||||||
"decamelize": {
|
|
||||||
"version": "1.0.0",
|
|
||||||
"from": "decamelize@>=1.0.0 <2.0.0",
|
|
||||||
"resolved": "https://registry.npmjs.org/decamelize/-/decamelize-1.0.0.tgz"
|
|
||||||
},
|
|
||||||
"deep-is": {
|
|
||||||
"version": "0.1.3",
|
|
||||||
"from": "deep-is@>=0.1.2 <0.2.0",
|
|
||||||
"resolved": "https://registry.npmjs.org/deep-is/-/deep-is-0.1.3.tgz"
|
|
||||||
},
|
|
||||||
"esprima": {
|
|
||||||
"version": "1.1.1",
|
|
||||||
"from": "esprima@>=1.1.1 <1.2.0",
|
|
||||||
"resolved": "https://registry.npmjs.org/esprima/-/esprima-1.1.1.tgz"
|
|
||||||
},
|
|
||||||
"estraverse": {
|
|
||||||
"version": "1.5.1",
|
|
||||||
"from": "estraverse@>=1.5.1 <1.6.0",
|
|
||||||
"resolved": "https://registry.npmjs.org/estraverse/-/estraverse-1.5.1.tgz"
|
|
||||||
},
|
|
||||||
"esutils": {
|
|
||||||
"version": "1.0.0",
|
|
||||||
"from": "esutils@>=1.0.0 <1.1.0",
|
|
||||||
"resolved": "https://registry.npmjs.org/esutils/-/esutils-1.0.0.tgz"
|
|
||||||
},
|
|
||||||
"fast-levenshtein": {
|
|
||||||
"version": "1.0.7",
|
|
||||||
"from": "fast-levenshtein@>=1.0.0 <1.1.0",
|
|
||||||
"resolved": "https://registry.npmjs.org/fast-levenshtein/-/fast-levenshtein-1.0.7.tgz"
|
|
||||||
},
|
|
||||||
"levn": {
|
|
||||||
"version": "0.2.5",
|
|
||||||
"from": "levn@>=0.2.5 <0.3.0",
|
|
||||||
"resolved": "https://registry.npmjs.org/levn/-/levn-0.2.5.tgz"
|
|
||||||
},
|
|
||||||
"nopt": {
|
|
||||||
"version": "2.1.2",
|
|
||||||
"from": "nopt@>=2.1.2 <2.2.0",
|
|
||||||
"resolved": "https://registry.npmjs.org/nopt/-/nopt-2.1.2.tgz"
|
|
||||||
},
|
|
||||||
"optionator": {
|
|
||||||
"version": "0.5.0",
|
|
||||||
"from": "optionator@>=0.5.0 <0.6.0",
|
|
||||||
"resolved": "https://registry.npmjs.org/optionator/-/optionator-0.5.0.tgz"
|
|
||||||
},
|
|
||||||
"prelude-ls": {
|
|
||||||
"version": "1.1.2",
|
|
||||||
"from": "prelude-ls@>=1.1.1 <1.2.0",
|
|
||||||
"resolved": "https://registry.npmjs.org/prelude-ls/-/prelude-ls-1.1.2.tgz"
|
|
||||||
},
|
|
||||||
"reflect": {
|
|
||||||
"version": "0.1.3",
|
|
||||||
"from": "git://github.com/zaach/reflect.js.git",
|
|
||||||
"resolved": "git://github.com/zaach/reflect.js.git#286bcd79661c96ecc404357d3c0e35fdb54a6967"
|
|
||||||
},
|
|
||||||
"source-map": {
|
|
||||||
"version": "0.5.1",
|
|
||||||
"from": "source-map@>=0.5.1 <0.6.0",
|
|
||||||
"resolved": "https://registry.npmjs.org/source-map/-/source-map-0.5.1.tgz"
|
|
||||||
},
|
|
||||||
"type-check": {
|
|
||||||
"version": "0.3.1",
|
|
||||||
"from": "type-check@>=0.3.1 <0.4.0",
|
|
||||||
"resolved": "https://registry.npmjs.org/type-check/-/type-check-0.3.1.tgz"
|
|
||||||
},
|
|
||||||
"uglify-js": {
|
|
||||||
"version": "2.4.24",
|
|
||||||
"from": "git://github.com/mishoo/UglifyJS2.git",
|
|
||||||
"resolved": "git://github.com/mishoo/UglifyJS2.git#2a06c7758e24a64740473c8031eafbb7fefa213f",
|
|
||||||
"dependencies": {
|
|
||||||
"source-map": {
|
|
||||||
"version": "0.1.34",
|
|
||||||
"from": "source-map@0.1.34",
|
|
||||||
"resolved": "https://registry.npmjs.org/source-map/-/source-map-0.1.34.tgz"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"uglify-to-browserify": {
|
|
||||||
"version": "1.0.2",
|
|
||||||
"from": "uglify-to-browserify@>=1.0.0 <1.1.0",
|
|
||||||
"resolved": "https://registry.npmjs.org/uglify-to-browserify/-/uglify-to-browserify-1.0.2.tgz"
|
|
||||||
},
|
|
||||||
"window-size": {
|
|
||||||
"version": "0.1.0",
|
|
||||||
"from": "window-size@0.1.0",
|
|
||||||
"resolved": "https://registry.npmjs.org/window-size/-/window-size-0.1.0.tgz"
|
|
||||||
},
|
|
||||||
"wordwrap": {
|
|
||||||
"version": "0.0.2",
|
|
||||||
"from": "wordwrap@0.0.2",
|
|
||||||
"resolved": "https://registry.npmjs.org/wordwrap/-/wordwrap-0.0.2.tgz"
|
|
||||||
},
|
|
||||||
"yargs": {
|
|
||||||
"version": "3.10.0",
|
|
||||||
"from": "yargs@>=3.10.0 <3.11.0",
|
|
||||||
"resolved": "https://registry.npmjs.org/yargs/-/yargs-3.10.0.tgz"
|
|
||||||
},
|
|
||||||
"zeparser": {
|
|
||||||
"version": "0.0.7",
|
|
||||||
"from": "git://github.com/qfox/ZeParser.git",
|
|
||||||
"resolved": "git://github.com/qfox/ZeParser.git#c99240c5ba7054c467733800ff38265958a2dda9"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -4,7 +4,7 @@
|
|||||||
"homepage": "http://lisperator.net/uglifyjs",
|
"homepage": "http://lisperator.net/uglifyjs",
|
||||||
"author": "Mihai Bazon <mihai.bazon@gmail.com> (http://lisperator.net/)",
|
"author": "Mihai Bazon <mihai.bazon@gmail.com> (http://lisperator.net/)",
|
||||||
"license": "BSD-2-Clause",
|
"license": "BSD-2-Clause",
|
||||||
"version": "2.6.4",
|
"version": "2.8.0",
|
||||||
"engines": {
|
"engines": {
|
||||||
"node": ">=0.8.0"
|
"node": ">=0.8.0"
|
||||||
},
|
},
|
||||||
|
|||||||
49
test/benchmark.js
Normal file
49
test/benchmark.js
Normal file
@@ -0,0 +1,49 @@
|
|||||||
|
#! /usr/bin/env node
|
||||||
|
// -*- js -*-
|
||||||
|
|
||||||
|
"use strict";
|
||||||
|
|
||||||
|
var createHash = require("crypto").createHash;
|
||||||
|
var fork = require("child_process").fork;
|
||||||
|
var args = process.argv.slice(2);
|
||||||
|
if (!args.length) {
|
||||||
|
args.push("-mc", "warnings=false");
|
||||||
|
}
|
||||||
|
args.push("--stats");
|
||||||
|
var urls = [
|
||||||
|
"https://code.jquery.com/jquery-3.1.1.js",
|
||||||
|
"https://code.angularjs.org/1.6.1/angular.js",
|
||||||
|
"https://cdnjs.cloudflare.com/ajax/libs/mathjs/3.9.0/math.js",
|
||||||
|
"https://maxcdn.bootstrapcdn.com/bootstrap/3.3.7/js/bootstrap.js",
|
||||||
|
"https://unpkg.com/react@15.3.2/dist/react.js",
|
||||||
|
"http://builds.emberjs.com/tags/v2.11.0/ember.prod.js",
|
||||||
|
"https://cdn.jsdelivr.net/lodash/4.17.4/lodash.js",
|
||||||
|
"https://cdnjs.cloudflare.com/ajax/libs/d3/4.5.0/d3.js",
|
||||||
|
];
|
||||||
|
var results = {};
|
||||||
|
var remaining = 2 * urls.length;
|
||||||
|
function done() {
|
||||||
|
if (!--remaining) {
|
||||||
|
urls.forEach(function(url) {
|
||||||
|
console.log();
|
||||||
|
console.log(url);
|
||||||
|
console.log(results[url].time);
|
||||||
|
console.log("SHA1:", results[url].sha1);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
urls.forEach(function(url) {
|
||||||
|
results[url] = { time: "" };
|
||||||
|
require(url.slice(0, url.indexOf(":"))).get(url, function(res) {
|
||||||
|
var uglifyjs = fork("bin/uglifyjs", args, { silent: true });
|
||||||
|
res.pipe(uglifyjs.stdin);
|
||||||
|
uglifyjs.stdout.pipe(createHash("sha1")).on("data", function(data) {
|
||||||
|
results[url].sha1 = data.toString("hex");
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
uglifyjs.stderr.setEncoding("utf8");
|
||||||
|
uglifyjs.stderr.on("data", function(data) {
|
||||||
|
results[url].time += data;
|
||||||
|
}).on("end", done)
|
||||||
|
});
|
||||||
|
});
|
||||||
@@ -21,10 +21,19 @@ constant_join: {
|
|||||||
input: {
|
input: {
|
||||||
var a = [ "foo", "bar", "baz" ].join("");
|
var a = [ "foo", "bar", "baz" ].join("");
|
||||||
var a1 = [ "foo", "bar", "baz" ].join();
|
var a1 = [ "foo", "bar", "baz" ].join();
|
||||||
|
var a2 = [ "foo", "bar", "baz" ].join(null);
|
||||||
|
var a3 = [ "foo", "bar", "baz" ].join(void 0);
|
||||||
|
var a4 = [ "foo", , "baz" ].join();
|
||||||
|
var a5 = [ "foo", null, "baz" ].join();
|
||||||
|
var a6 = [ "foo", void 0, "baz" ].join();
|
||||||
var b = [ "foo", 1, 2, 3, "bar" ].join("");
|
var b = [ "foo", 1, 2, 3, "bar" ].join("");
|
||||||
var c = [ boo(), "foo", 1, 2, 3, "bar", bar() ].join("");
|
var c = [ boo(), "foo", 1, 2, 3, "bar", bar() ].join("");
|
||||||
var c1 = [ boo(), bar(), "foo", 1, 2, 3, "bar", bar() ].join("");
|
var c1 = [ boo(), bar(), "foo", 1, 2, 3, "bar", bar() ].join("");
|
||||||
var c2 = [ 1, 2, "foo", "bar", baz() ].join("");
|
var c2 = [ 1, 2, "foo", "bar", baz() ].join("");
|
||||||
|
var c3 = [ boo() + bar() + "foo", 1, 2, 3, "bar", bar() + "foo" ].join("");
|
||||||
|
var c4 = [ 1, 2, null, undefined, "foo", "bar", baz() ].join("");
|
||||||
|
var c5 = [ boo() + bar() + "foo", 1, 2, 3, "bar", bar() + "foo" ].join();
|
||||||
|
var c6 = [ 1, 2, null, undefined, "foo", "bar", baz() ].join();
|
||||||
var d = [ "foo", 1 + 2 + "bar", "baz" ].join("-");
|
var d = [ "foo", 1 + 2 + "bar", "baz" ].join("-");
|
||||||
var e = [].join(foo + bar);
|
var e = [].join(foo + bar);
|
||||||
var f = [].join("");
|
var f = [].join("");
|
||||||
@@ -33,10 +42,19 @@ constant_join: {
|
|||||||
expect: {
|
expect: {
|
||||||
var a = "foobarbaz";
|
var a = "foobarbaz";
|
||||||
var a1 = "foo,bar,baz";
|
var a1 = "foo,bar,baz";
|
||||||
|
var a2 = "foonullbarnullbaz";
|
||||||
|
var a3 = "foo,bar,baz";
|
||||||
|
var a4 = "foo,,baz";
|
||||||
|
var a5 = "foo,,baz";
|
||||||
|
var a6 = "foo,,baz";
|
||||||
var b = "foo123bar";
|
var b = "foo123bar";
|
||||||
var c = boo() + "foo123bar" + bar();
|
var c = boo() + "foo123bar" + bar();
|
||||||
var c1 = "" + boo() + bar() + "foo123bar" + bar();
|
var c1 = "" + boo() + bar() + "foo123bar" + bar();
|
||||||
var c2 = "12foobar" + baz();
|
var c2 = "12foobar" + baz();
|
||||||
|
var c3 = boo() + bar() + "foo123bar" + bar() + "foo";
|
||||||
|
var c4 = "12foobar" + baz();
|
||||||
|
var c5 = [ boo() + bar() + "foo", 1, 2, 3, "bar", bar() + "foo" ].join();
|
||||||
|
var c6 = [ "1,2,,,foo,bar", baz() ].join();
|
||||||
var d = "foo-3bar-baz";
|
var d = "foo-3bar-baz";
|
||||||
var e = [].join(foo + bar);
|
var e = [].join(foo + bar);
|
||||||
var f = "";
|
var f = "";
|
||||||
@@ -72,3 +90,88 @@ constant_join_2: {
|
|||||||
var f = "strstr" + variable + "foobarmoo" + foo;
|
var f = "strstr" + variable + "foobarmoo" + foo;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
constant_join_3: {
|
||||||
|
options = {
|
||||||
|
unsafe: true,
|
||||||
|
evaluate: true,
|
||||||
|
};
|
||||||
|
input: {
|
||||||
|
var a = [ null ].join();
|
||||||
|
var b = [ , ].join();
|
||||||
|
var c = [ , 1, , 3 ].join();
|
||||||
|
var d = [ foo ].join();
|
||||||
|
var e = [ foo, null, undefined, bar ].join("-");
|
||||||
|
var f = [ foo, bar ].join("");
|
||||||
|
var g = [ null, "foo", null, bar + "baz" ].join("");
|
||||||
|
var h = [ null, "foo", null, bar + "baz" ].join("-");
|
||||||
|
var i = [ "foo" + bar, null, baz + "moo" ].join("");
|
||||||
|
var j = [ foo + "bar", baz ].join("");
|
||||||
|
var k = [ foo, "bar" + baz ].join("");
|
||||||
|
var l = [ foo, bar + "baz" ].join("");
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
var a = "";
|
||||||
|
var b = "";
|
||||||
|
var c = ",1,,3";
|
||||||
|
var d = "" + foo;
|
||||||
|
var e = [ foo, "-", bar ].join("-");
|
||||||
|
var f = "" + foo + bar;
|
||||||
|
var g = "foo" + bar + "baz";
|
||||||
|
var h = [ "-foo-", bar + "baz" ].join("-");
|
||||||
|
var i = "foo" + bar + baz + "moo";
|
||||||
|
var j = foo + "bar" + baz;
|
||||||
|
var k = foo + "bar" + baz;
|
||||||
|
var l = foo + (bar + "baz");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
for_loop: {
|
||||||
|
options = {
|
||||||
|
unsafe : true,
|
||||||
|
unused : true,
|
||||||
|
evaluate : true,
|
||||||
|
reduce_vars : true
|
||||||
|
};
|
||||||
|
input: {
|
||||||
|
function f0() {
|
||||||
|
var a = [1, 2, 3];
|
||||||
|
for (var i = 0; i < a.length; i++) {
|
||||||
|
console.log(a[i]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function f1() {
|
||||||
|
var a = [1, 2, 3];
|
||||||
|
for (var i = 0, len = a.length; i < len; i++) {
|
||||||
|
console.log(a[i]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function f2() {
|
||||||
|
var a = [1, 2, 3];
|
||||||
|
for (var i = 0; i < a.length; i++) {
|
||||||
|
a[i]++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
function f0() {
|
||||||
|
var a = [1, 2, 3];
|
||||||
|
for (var i = 0; i < 3; i++)
|
||||||
|
console.log(a[i]);
|
||||||
|
}
|
||||||
|
|
||||||
|
function f1() {
|
||||||
|
var a = [1, 2, 3];
|
||||||
|
for (var i = 0; i < 3; i++)
|
||||||
|
console.log(a[i]);
|
||||||
|
}
|
||||||
|
|
||||||
|
function f2() {
|
||||||
|
var a = [1, 2, 3];
|
||||||
|
for (var i = 0; i < a.length; i++)
|
||||||
|
a[i]++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|||||||
@@ -2,6 +2,7 @@ ascii_only_true: {
|
|||||||
options = {}
|
options = {}
|
||||||
beautify = {
|
beautify = {
|
||||||
ascii_only : true,
|
ascii_only : true,
|
||||||
|
screw_ie8 : true,
|
||||||
beautify : false,
|
beautify : false,
|
||||||
}
|
}
|
||||||
input: {
|
input: {
|
||||||
@@ -12,13 +13,14 @@ ascii_only_true: {
|
|||||||
"\x20\x21\x22\x23 ... \x7d\x7e\x7f\x80\x81 ... \xfe\xff\u0fff\uffff";
|
"\x20\x21\x22\x23 ... \x7d\x7e\x7f\x80\x81 ... \xfe\xff\u0fff\uffff";
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
expect_exact: 'function f(){return"\\x000\\x001\\x007\\08\\0"+"\\0\\x01\\x02\\x03\\x04\\x05\\x06\\x07\\b\\t\\n\\x0B\\f\\r\\x0e\\x0f"+"\\x10\\x11\\x12\\x13\\x14\\x15\\x16\\x17\\x18\\x19\\x1a\\x1b\\x1c\\x1d\\x1e\\x1f"+\' !"# ... }~\\x7f\\x80\\x81 ... \\xfe\\xff\\u0fff\\uffff\'}'
|
expect_exact: 'function f(){return"\\x000\\x001\\x007\\08\\0"+"\\0\\x01\\x02\\x03\\x04\\x05\\x06\\x07\\b\\t\\n\\v\\f\\r\\x0e\\x0f"+"\\x10\\x11\\x12\\x13\\x14\\x15\\x16\\x17\\x18\\x19\\x1a\\x1b\\x1c\\x1d\\x1e\\x1f"+\' !"# ... }~\\x7f\\x80\\x81 ... \\xfe\\xff\\u0fff\\uffff\'}'
|
||||||
}
|
}
|
||||||
|
|
||||||
ascii_only_false: {
|
ascii_only_false: {
|
||||||
options = {}
|
options = {}
|
||||||
beautify = {
|
beautify = {
|
||||||
ascii_only : false,
|
ascii_only : false,
|
||||||
|
screw_ie8 : true,
|
||||||
beautify : false,
|
beautify : false,
|
||||||
}
|
}
|
||||||
input: {
|
input: {
|
||||||
@@ -29,6 +31,6 @@ ascii_only_false: {
|
|||||||
"\x20\x21\x22\x23 ... \x7d\x7e\x7f\x80\x81 ... \xfe\xff\u0fff\uffff";
|
"\x20\x21\x22\x23 ... \x7d\x7e\x7f\x80\x81 ... \xfe\xff\u0fff\uffff";
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
expect_exact: 'function f(){return"\\x000\\x001\\x007\\08\\0"+"\\0\x01\x02\x03\x04\x05\x06\x07\\b\\t\\n\\x0B\\f\\r\x0e\x0f"+"\x10\x11\x12\x13\x14\x15\x16\x17\x18\x19\x1a\x1b\x1c\x1d\x1e\x1f"+\' !"# ... }~\x7f\x80\x81 ... \xfe\xff\u0fff\uffff\'}'
|
expect_exact: 'function f(){return"\\x000\\x001\\x007\\08\\0"+"\\0\x01\x02\x03\x04\x05\x06\x07\\b\\t\\n\\v\\f\\r\x0e\x0f"+"\x10\x11\x12\x13\x14\x15\x16\x17\x18\x19\x1a\x1b\x1c\x1d\x1e\x1f"+\' !"# ... }~\x7f\x80\x81 ... \xfe\xff\u0fff\uffff\'}'
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -90,13 +90,13 @@ asm_mixed: {
|
|||||||
}
|
}
|
||||||
function no_asm_GeometricMean(stdlib, foreign, buffer) {
|
function no_asm_GeometricMean(stdlib, foreign, buffer) {
|
||||||
function logSum(start, end) {
|
function logSum(start, end) {
|
||||||
start = 0 | start, end = 0 | end;
|
start |= 0, end |= 0;
|
||||||
var sum = 0, p = 0, q = 0;
|
var sum = 0, p = 0, q = 0;
|
||||||
for (p = start << 3, q = end << 3; (0 | p) < (0 | q); p = p + 8 | 0) sum += +log(values[p >> 3]);
|
for (p = start << 3, q = end << 3; (0 | p) < (0 | q); p = p + 8 | 0) sum += +log(values[p >> 3]);
|
||||||
return +sum;
|
return +sum;
|
||||||
}
|
}
|
||||||
function geometricMean(start, end) {
|
function geometricMean(start, end) {
|
||||||
return start = 0 | start, end = 0 | end, +exp(+logSum(start, end) / +(end - start | 0));
|
return start |= 0, end |= 0, +exp(+logSum(start, end) / +(end - start | 0));
|
||||||
}
|
}
|
||||||
var exp = stdlib.Math.exp, log = stdlib.Math.log, values = new stdlib.Float64Array(buffer);
|
var exp = stdlib.Math.exp, log = stdlib.Math.log, values = new stdlib.Float64Array(buffer);
|
||||||
return { geometricMean: geometricMean };
|
return { geometricMean: geometricMean };
|
||||||
|
|||||||
238
test/compress/assignment.js
Normal file
238
test/compress/assignment.js
Normal file
@@ -0,0 +1,238 @@
|
|||||||
|
op_equals_left_local_var: {
|
||||||
|
options = {
|
||||||
|
evaluate: true,
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
var x;
|
||||||
|
|
||||||
|
x = x + 3;
|
||||||
|
x = x - 3;
|
||||||
|
x = x / 3;
|
||||||
|
x = x * 3;
|
||||||
|
x = x >> 3;
|
||||||
|
x = x << 3;
|
||||||
|
x = x >>> 3;
|
||||||
|
x = x | 3;
|
||||||
|
x = x ^ 3;
|
||||||
|
x = x % 3;
|
||||||
|
x = x & 3;
|
||||||
|
|
||||||
|
x = x + g();
|
||||||
|
x = x - g();
|
||||||
|
x = x / g();
|
||||||
|
x = x * g();
|
||||||
|
x = x >> g();
|
||||||
|
x = x << g();
|
||||||
|
x = x >>> g();
|
||||||
|
x = x | g();
|
||||||
|
x = x ^ g();
|
||||||
|
x = x % g();
|
||||||
|
x = x & g();
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
var x;
|
||||||
|
|
||||||
|
x += 3;
|
||||||
|
x -= 3;
|
||||||
|
x /= 3;
|
||||||
|
x *= 3;
|
||||||
|
x >>= 3;
|
||||||
|
x <<= 3;
|
||||||
|
x >>>= 3;
|
||||||
|
x |= 3;
|
||||||
|
x ^= 3;
|
||||||
|
x %= 3;
|
||||||
|
x &= 3;
|
||||||
|
|
||||||
|
x += g();
|
||||||
|
x -= g();
|
||||||
|
x /= g();
|
||||||
|
x *= g();
|
||||||
|
x >>= g();
|
||||||
|
x <<= g();
|
||||||
|
x >>>= g();
|
||||||
|
x |= g();
|
||||||
|
x ^= g();
|
||||||
|
x %= g();
|
||||||
|
x &= g();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
op_equals_right_local_var: {
|
||||||
|
options = {
|
||||||
|
evaluate: true,
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
var x;
|
||||||
|
|
||||||
|
x = (x -= 2) ^ x;
|
||||||
|
|
||||||
|
x = 3 + x;
|
||||||
|
x = 3 - x;
|
||||||
|
x = 3 / x;
|
||||||
|
x = 3 * x;
|
||||||
|
x = 3 >> x;
|
||||||
|
x = 3 << x;
|
||||||
|
x = 3 >>> x;
|
||||||
|
x = 3 | x;
|
||||||
|
x = 3 ^ x;
|
||||||
|
x = 3 % x;
|
||||||
|
x = 3 & x;
|
||||||
|
|
||||||
|
x = g() + x;
|
||||||
|
x = g() - x;
|
||||||
|
x = g() / x;
|
||||||
|
x = g() * x;
|
||||||
|
x = g() >> x;
|
||||||
|
x = g() << x;
|
||||||
|
x = g() >>> x;
|
||||||
|
x = g() | x;
|
||||||
|
x = g() ^ x;
|
||||||
|
x = g() % x;
|
||||||
|
x = g() & x;
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
var x;
|
||||||
|
|
||||||
|
x = (x -= 2) ^ x;
|
||||||
|
|
||||||
|
x = 3 + x;
|
||||||
|
x = 3 - x;
|
||||||
|
x = 3 / x;
|
||||||
|
x *= 3;
|
||||||
|
x = 3 >> x;
|
||||||
|
x = 3 << x;
|
||||||
|
x = 3 >>> x;
|
||||||
|
x |= 3;
|
||||||
|
x ^= 3;
|
||||||
|
x = 3 % x;
|
||||||
|
x &= 3;
|
||||||
|
|
||||||
|
x = g() + x;
|
||||||
|
x = g() - x;
|
||||||
|
x = g() / x;
|
||||||
|
x = g() * x;
|
||||||
|
x = g() >> x;
|
||||||
|
x = g() << x;
|
||||||
|
x = g() >>> x;
|
||||||
|
x = g() | x;
|
||||||
|
x = g() ^ x;
|
||||||
|
x = g() % x;
|
||||||
|
x = g() & x;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
op_equals_left_global_var: {
|
||||||
|
options = {
|
||||||
|
evaluate: true,
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
x = x + 3;
|
||||||
|
x = x - 3;
|
||||||
|
x = x / 3;
|
||||||
|
x = x * 3;
|
||||||
|
x = x >> 3;
|
||||||
|
x = x << 3;
|
||||||
|
x = x >>> 3;
|
||||||
|
x = x | 3;
|
||||||
|
x = x ^ 3;
|
||||||
|
x = x % 3;
|
||||||
|
x = x & 3;
|
||||||
|
|
||||||
|
x = x + g();
|
||||||
|
x = x - g();
|
||||||
|
x = x / g();
|
||||||
|
x = x * g();
|
||||||
|
x = x >> g();
|
||||||
|
x = x << g();
|
||||||
|
x = x >>> g();
|
||||||
|
x = x | g();
|
||||||
|
x = x ^ g();
|
||||||
|
x = x % g();
|
||||||
|
x = x & g();
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
x += 3;
|
||||||
|
x -= 3;
|
||||||
|
x /= 3;
|
||||||
|
x *= 3;
|
||||||
|
x >>= 3;
|
||||||
|
x <<= 3;
|
||||||
|
x >>>= 3;
|
||||||
|
x |= 3;
|
||||||
|
x ^= 3;
|
||||||
|
x %= 3;
|
||||||
|
x &= 3;
|
||||||
|
|
||||||
|
x += g();
|
||||||
|
x -= g();
|
||||||
|
x /= g();
|
||||||
|
x *= g();
|
||||||
|
x >>= g();
|
||||||
|
x <<= g();
|
||||||
|
x >>>= g();
|
||||||
|
x |= g();
|
||||||
|
x ^= g();
|
||||||
|
x %= g();
|
||||||
|
x &= g();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
op_equals_right_global_var: {
|
||||||
|
options = {
|
||||||
|
evaluate: true,
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
x = (x -= 2) ^ x;
|
||||||
|
|
||||||
|
x = 3 + x;
|
||||||
|
x = 3 - x;
|
||||||
|
x = 3 / x;
|
||||||
|
x = 3 * x;
|
||||||
|
x = 3 >> x;
|
||||||
|
x = 3 << x;
|
||||||
|
x = 3 >>> x;
|
||||||
|
x = 3 | x;
|
||||||
|
x = 3 ^ x;
|
||||||
|
x = 3 % x;
|
||||||
|
x = 3 & x;
|
||||||
|
|
||||||
|
x = g() + x;
|
||||||
|
x = g() - x;
|
||||||
|
x = g() / x;
|
||||||
|
x = g() * x;
|
||||||
|
x = g() >> x;
|
||||||
|
x = g() << x;
|
||||||
|
x = g() >>> x;
|
||||||
|
x = g() | x;
|
||||||
|
x = g() ^ x;
|
||||||
|
x = g() % x;
|
||||||
|
x = g() & x;
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
x = (x -= 2) ^ x;
|
||||||
|
|
||||||
|
x = 3 + x;
|
||||||
|
x = 3 - x;
|
||||||
|
x = 3 / x;
|
||||||
|
x *= 3;
|
||||||
|
x = 3 >> x;
|
||||||
|
x = 3 << x;
|
||||||
|
x = 3 >>> x;
|
||||||
|
x |= 3;
|
||||||
|
x ^= 3;
|
||||||
|
x = 3 % x;
|
||||||
|
x &= 3;
|
||||||
|
|
||||||
|
x = g() + x;
|
||||||
|
x = g() - x;
|
||||||
|
x = g() / x;
|
||||||
|
x = g() * x;
|
||||||
|
x = g() >> x;
|
||||||
|
x = g() << x;
|
||||||
|
x = g() >>> x;
|
||||||
|
x = g() | x;
|
||||||
|
x = g() ^ x;
|
||||||
|
x = g() % x;
|
||||||
|
x = g() & x;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -338,8 +338,9 @@ collapse_vars_while: {
|
|||||||
collapse_vars_do_while: {
|
collapse_vars_do_while: {
|
||||||
options = {
|
options = {
|
||||||
collapse_vars:true, sequences:true, properties:true, dead_code:true, conditionals:true,
|
collapse_vars:true, sequences:true, properties:true, dead_code:true, conditionals:true,
|
||||||
comparisons:true, evaluate:true, booleans:false, loops:false, unused:true, hoist_funs:true,
|
comparisons:true, evaluate:true, booleans:false, loops:false, unused:"keep_assign",
|
||||||
keep_fargs:true, if_return:true, join_vars:true, cascade:true, side_effects:true
|
hoist_funs:true, keep_fargs:true, if_return:true, join_vars:true, cascade:true,
|
||||||
|
side_effects:true
|
||||||
}
|
}
|
||||||
input: {
|
input: {
|
||||||
function f1(y) {
|
function f1(y) {
|
||||||
@@ -409,6 +410,79 @@ collapse_vars_do_while: {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
collapse_vars_do_while_drop_assign: {
|
||||||
|
options = {
|
||||||
|
collapse_vars:true, sequences:true, properties:true, dead_code:true, conditionals:true,
|
||||||
|
comparisons:true, evaluate:true, booleans:false, loops:false, unused:true, hoist_funs:true,
|
||||||
|
keep_fargs:true, if_return:true, join_vars:true, cascade:true, side_effects:true
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
function f1(y) {
|
||||||
|
// The constant do-while condition `c` will be replaced.
|
||||||
|
var c = 9;
|
||||||
|
do { } while (c === 77);
|
||||||
|
}
|
||||||
|
function f2(y) {
|
||||||
|
// The non-constant do-while condition `c` will not be replaced.
|
||||||
|
var c = 5 - y;
|
||||||
|
do { } while (c);
|
||||||
|
}
|
||||||
|
function f3(y) {
|
||||||
|
// The constant `x` will be replaced in the do loop body.
|
||||||
|
function fn(n) { console.log(n); }
|
||||||
|
var a = 2, x = 7;
|
||||||
|
do {
|
||||||
|
fn(a = x);
|
||||||
|
break;
|
||||||
|
} while (y);
|
||||||
|
}
|
||||||
|
function f4(y) {
|
||||||
|
// The non-constant `a` will not be replaced in the do loop body.
|
||||||
|
var a = y / 4;
|
||||||
|
do {
|
||||||
|
return a;
|
||||||
|
} while (y);
|
||||||
|
}
|
||||||
|
function f5(y) {
|
||||||
|
function p(x) { console.log(x); }
|
||||||
|
do {
|
||||||
|
// The non-constant `a` will be replaced in p(a)
|
||||||
|
// because it is declared in same block.
|
||||||
|
var a = y - 3;
|
||||||
|
p(a);
|
||||||
|
} while (--y);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
function f1(y) {
|
||||||
|
do ; while (false);
|
||||||
|
}
|
||||||
|
function f2(y) {
|
||||||
|
var c = 5 - y;
|
||||||
|
do ; while (c);
|
||||||
|
}
|
||||||
|
function f3(y) {
|
||||||
|
function fn(n) { console.log(n); }
|
||||||
|
do {
|
||||||
|
fn(7);
|
||||||
|
break;
|
||||||
|
} while (y);
|
||||||
|
}
|
||||||
|
function f4(y) {
|
||||||
|
var a = y / 4;
|
||||||
|
do
|
||||||
|
return a;
|
||||||
|
while (y);
|
||||||
|
}
|
||||||
|
function f5(y) {
|
||||||
|
function p(x) { console.log(x); }
|
||||||
|
do {
|
||||||
|
p(y - 3);
|
||||||
|
} while (--y);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
collapse_vars_seq: {
|
collapse_vars_seq: {
|
||||||
options = {
|
options = {
|
||||||
collapse_vars:true, sequences:true, properties:true, dead_code:true, conditionals:true,
|
collapse_vars:true, sequences:true, properties:true, dead_code:true, conditionals:true,
|
||||||
@@ -567,8 +641,9 @@ collapse_vars_assignment: {
|
|||||||
collapse_vars_lvalues: {
|
collapse_vars_lvalues: {
|
||||||
options = {
|
options = {
|
||||||
collapse_vars:true, sequences:true, properties:true, dead_code:true, conditionals:true,
|
collapse_vars:true, sequences:true, properties:true, dead_code:true, conditionals:true,
|
||||||
comparisons:true, evaluate:true, booleans:true, loops:true, unused:true, hoist_funs:true,
|
comparisons:true, evaluate:true, booleans:true, loops:true, unused:"keep_assign",
|
||||||
keep_fargs:true, if_return:true, join_vars:true, cascade:true, side_effects:true
|
hoist_funs:true, keep_fargs:true, if_return:true, join_vars:true, cascade:true,
|
||||||
|
side_effects:true
|
||||||
}
|
}
|
||||||
input: {
|
input: {
|
||||||
function f0(x) { var i = ++x; return x += i; }
|
function f0(x) { var i = ++x; return x += i; }
|
||||||
@@ -593,7 +668,38 @@ collapse_vars_lvalues: {
|
|||||||
function f7(x) { var w = e1(), v = e2(), c = v - x; return (w = x) - c; }
|
function f7(x) { var w = e1(), v = e2(), c = v - x; return (w = x) - c; }
|
||||||
function f8(x) { var w = e1(), v = e2(); return (w = x) - (v - x); }
|
function f8(x) { var w = e1(), v = e2(); return (w = x) - (v - x); }
|
||||||
function f9(x) { var w = e1(); return e2() - x - (w = x); }
|
function f9(x) { var w = e1(); return e2() - x - (w = x); }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
collapse_vars_lvalues_drop_assign: {
|
||||||
|
options = {
|
||||||
|
collapse_vars:true, sequences:true, properties:true, dead_code:true, conditionals:true,
|
||||||
|
comparisons:true, evaluate:true, booleans:true, loops:true, unused:true, hoist_funs:true,
|
||||||
|
keep_fargs:true, if_return:true, join_vars:true, cascade:true, side_effects:true
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
function f0(x) { var i = ++x; return x += i; }
|
||||||
|
function f1(x) { var a = (x -= 3); return x += a; }
|
||||||
|
function f2(x) { var z = x, a = ++z; return z += a; }
|
||||||
|
function f3(x) { var a = (x -= 3), b = x + a; return b; }
|
||||||
|
function f4(x) { var a = (x -= 3); return x + a; }
|
||||||
|
function f5(x) { var w = e1(), v = e2(), c = v = --x, b = w = x; return b - c; }
|
||||||
|
function f6(x) { var w = e1(), v = e2(), c = v = --x, b = w = x; return c - b; }
|
||||||
|
function f7(x) { var w = e1(), v = e2(), c = v - x, b = w = x; return b - c; }
|
||||||
|
function f8(x) { var w = e1(), v = e2(), b = w = x, c = v - x; return b - c; }
|
||||||
|
function f9(x) { var w = e1(), v = e2(), b = w = x, c = v - x; return c - b; }
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
function f0(x) { var i = ++x; return x += i; }
|
||||||
|
function f1(x) { var a = (x -= 3); return x += a; }
|
||||||
|
function f2(x) { var z = x, a = ++z; return z += a; }
|
||||||
|
function f3(x) { var a = (x -= 3); return x + a; }
|
||||||
|
function f4(x) { var a = (x -= 3); return x + a; }
|
||||||
|
function f5(x) { var v = (e1(), e2()), c = v = --x; return x - c; }
|
||||||
|
function f6(x) { e1(), e2(); return --x - x; }
|
||||||
|
function f7(x) { var v = (e1(), e2()), c = v - x; return x - c; }
|
||||||
|
function f8(x) { var v = (e1(), e2()); return x - (v - x); }
|
||||||
|
function f9(x) { e1(); return e2() - x - x; }
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -24,3 +24,143 @@ concat_1: {
|
|||||||
var f = "\x00360\08\0";
|
var f = "\x00360\08\0";
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
concat_2: {
|
||||||
|
options = {};
|
||||||
|
input: {
|
||||||
|
console.log(
|
||||||
|
1 + (2 + 3),
|
||||||
|
1 + (2 + "3"),
|
||||||
|
1 + ("2" + 3),
|
||||||
|
1 + ("2" + "3"),
|
||||||
|
"1" + (2 + 3),
|
||||||
|
"1" + (2 + "3"),
|
||||||
|
"1" + ("2" + 3),
|
||||||
|
"1" + ("2" + "3")
|
||||||
|
);
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
console.log(
|
||||||
|
1 + (2 + 3),
|
||||||
|
1 + (2 + "3"),
|
||||||
|
1 + "2" + 3,
|
||||||
|
1 + "2" + "3",
|
||||||
|
"1" + (2 + 3),
|
||||||
|
"1" + 2 + "3",
|
||||||
|
"1" + "2" + 3,
|
||||||
|
"1" + "2" + "3"
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
concat_3: {
|
||||||
|
options = {};
|
||||||
|
input: {
|
||||||
|
console.log(
|
||||||
|
1 + 2 + (3 + 4 + 5),
|
||||||
|
1 + 2 + (3 + 4 + "5"),
|
||||||
|
1 + 2 + (3 + "4" + 5),
|
||||||
|
1 + 2 + (3 + "4" + "5"),
|
||||||
|
1 + 2 + ("3" + 4 + 5),
|
||||||
|
1 + 2 + ("3" + 4 + "5"),
|
||||||
|
1 + 2 + ("3" + "4" + 5),
|
||||||
|
1 + 2 + ("3" + "4" + "5")
|
||||||
|
);
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
console.log(
|
||||||
|
1 + 2 + (3 + 4 + 5),
|
||||||
|
1 + 2 + (3 + 4 + "5"),
|
||||||
|
1 + 2 + (3 + "4") + 5,
|
||||||
|
1 + 2 + (3 + "4") + "5",
|
||||||
|
1 + 2 + "3" + 4 + 5,
|
||||||
|
1 + 2 + "3" + 4 + "5",
|
||||||
|
1 + 2 + "3" + "4" + 5,
|
||||||
|
1 + 2 + "3" + "4" + "5"
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
concat_4: {
|
||||||
|
options = {};
|
||||||
|
input: {
|
||||||
|
console.log(
|
||||||
|
1 + "2" + (3 + 4 + 5),
|
||||||
|
1 + "2" + (3 + 4 + "5"),
|
||||||
|
1 + "2" + (3 + "4" + 5),
|
||||||
|
1 + "2" + (3 + "4" + "5"),
|
||||||
|
1 + "2" + ("3" + 4 + 5),
|
||||||
|
1 + "2" + ("3" + 4 + "5"),
|
||||||
|
1 + "2" + ("3" + "4" + 5),
|
||||||
|
1 + "2" + ("3" + "4" + "5")
|
||||||
|
);
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
console.log(
|
||||||
|
1 + "2" + (3 + 4 + 5),
|
||||||
|
1 + "2" + (3 + 4) + "5",
|
||||||
|
1 + "2" + 3 + "4" + 5,
|
||||||
|
1 + "2" + 3 + "4" + "5",
|
||||||
|
1 + "2" + "3" + 4 + 5,
|
||||||
|
1 + "2" + "3" + 4 + "5",
|
||||||
|
1 + "2" + "3" + "4" + 5,
|
||||||
|
1 + "2" + "3" + "4" + "5"
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
concat_5: {
|
||||||
|
options = {};
|
||||||
|
input: {
|
||||||
|
console.log(
|
||||||
|
"1" + 2 + (3 + 4 + 5),
|
||||||
|
"1" + 2 + (3 + 4 + "5"),
|
||||||
|
"1" + 2 + (3 + "4" + 5),
|
||||||
|
"1" + 2 + (3 + "4" + "5"),
|
||||||
|
"1" + 2 + ("3" + 4 + 5),
|
||||||
|
"1" + 2 + ("3" + 4 + "5"),
|
||||||
|
"1" + 2 + ("3" + "4" + 5),
|
||||||
|
"1" + 2 + ("3" + "4" + "5")
|
||||||
|
);
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
console.log(
|
||||||
|
"1" + 2 + (3 + 4 + 5),
|
||||||
|
"1" + 2 + (3 + 4) + "5",
|
||||||
|
"1" + 2 + 3 + "4" + 5,
|
||||||
|
"1" + 2 + 3 + "4" + "5",
|
||||||
|
"1" + 2 + "3" + 4 + 5,
|
||||||
|
"1" + 2 + "3" + 4 + "5",
|
||||||
|
"1" + 2 + "3" + "4" + 5,
|
||||||
|
"1" + 2 + "3" + "4" + "5"
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
concat_6: {
|
||||||
|
options = {};
|
||||||
|
input: {
|
||||||
|
console.log(
|
||||||
|
"1" + "2" + (3 + 4 + 5),
|
||||||
|
"1" + "2" + (3 + 4 + "5"),
|
||||||
|
"1" + "2" + (3 + "4" + 5),
|
||||||
|
"1" + "2" + (3 + "4" + "5"),
|
||||||
|
"1" + "2" + ("3" + 4 + 5),
|
||||||
|
"1" + "2" + ("3" + 4 + "5"),
|
||||||
|
"1" + "2" + ("3" + "4" + 5),
|
||||||
|
"1" + "2" + ("3" + "4" + "5")
|
||||||
|
);
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
console.log(
|
||||||
|
"1" + "2" + (3 + 4 + 5),
|
||||||
|
"1" + "2" + (3 + 4) + "5",
|
||||||
|
"1" + "2" + 3 + "4" + 5,
|
||||||
|
"1" + "2" + 3 + "4" + "5",
|
||||||
|
"1" + "2" + "3" + 4 + 5,
|
||||||
|
"1" + "2" + "3" + 4 + "5",
|
||||||
|
"1" + "2" + "3" + "4" + 5,
|
||||||
|
"1" + "2" + "3" + "4" + "5"
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|||||||
@@ -635,166 +635,6 @@ ternary_boolean_alternative: {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
conditional_and: {
|
|
||||||
options = {
|
|
||||||
conditionals: true,
|
|
||||||
evaluate : true
|
|
||||||
};
|
|
||||||
input: {
|
|
||||||
var a;
|
|
||||||
// compress these
|
|
||||||
|
|
||||||
a = true && condition;
|
|
||||||
a = 1 && console.log("a");
|
|
||||||
a = 2 * 3 && 2 * condition;
|
|
||||||
a = 5 == 5 && condition + 3;
|
|
||||||
a = "string" && 4 - condition;
|
|
||||||
a = 5 + "" && condition / 5;
|
|
||||||
a = -4.5 && 6 << condition;
|
|
||||||
a = 6 && 7;
|
|
||||||
|
|
||||||
a = false && condition;
|
|
||||||
a = NaN && console.log("b");
|
|
||||||
a = 0 && console.log("c");
|
|
||||||
a = undefined && 2 * condition;
|
|
||||||
a = null && condition + 3;
|
|
||||||
a = 2 * 3 - 6 && 4 - condition;
|
|
||||||
a = 10 == 7 && condition / 5;
|
|
||||||
a = !"string" && 6 % condition;
|
|
||||||
a = 0 && 7;
|
|
||||||
|
|
||||||
// don't compress these
|
|
||||||
|
|
||||||
a = condition && true;
|
|
||||||
a = console.log("a") && 2;
|
|
||||||
a = 4 - condition && "string";
|
|
||||||
a = 6 << condition && -4.5;
|
|
||||||
|
|
||||||
a = condition && false;
|
|
||||||
a = console.log("b") && NaN;
|
|
||||||
a = console.log("c") && 0;
|
|
||||||
a = 2 * condition && undefined;
|
|
||||||
a = condition + 3 && null;
|
|
||||||
|
|
||||||
}
|
|
||||||
expect: {
|
|
||||||
var a;
|
|
||||||
|
|
||||||
a = condition;
|
|
||||||
a = console.log("a");
|
|
||||||
a = 2 * condition;
|
|
||||||
a = condition + 3;
|
|
||||||
a = 4 - condition;
|
|
||||||
a = condition / 5;
|
|
||||||
a = 6 << condition;
|
|
||||||
a = 7;
|
|
||||||
|
|
||||||
a = false;
|
|
||||||
a = NaN;
|
|
||||||
a = 0;
|
|
||||||
a = void 0;
|
|
||||||
a = null;
|
|
||||||
a = 0;
|
|
||||||
a = false;
|
|
||||||
a = false;
|
|
||||||
a = 0;
|
|
||||||
|
|
||||||
a = condition && true;
|
|
||||||
a = console.log("a") && 2;
|
|
||||||
a = 4 - condition && "string";
|
|
||||||
a = 6 << condition && -4.5;
|
|
||||||
|
|
||||||
a = condition && false;
|
|
||||||
a = console.log("b") && NaN;
|
|
||||||
a = console.log("c") && 0;
|
|
||||||
a = 2 * condition && void 0;
|
|
||||||
a = condition + 3 && null;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
conditional_or: {
|
|
||||||
options = {
|
|
||||||
conditionals: true,
|
|
||||||
evaluate : true
|
|
||||||
};
|
|
||||||
input: {
|
|
||||||
var a;
|
|
||||||
// compress these
|
|
||||||
|
|
||||||
a = true || condition;
|
|
||||||
a = 1 || console.log("a");
|
|
||||||
a = 2 * 3 || 2 * condition;
|
|
||||||
a = 5 == 5 || condition + 3;
|
|
||||||
a = "string" || 4 - condition;
|
|
||||||
a = 5 + "" || condition / 5;
|
|
||||||
a = -4.5 || 6 << condition;
|
|
||||||
a = 6 || 7;
|
|
||||||
|
|
||||||
a = false || condition;
|
|
||||||
a = 0 || console.log("b");
|
|
||||||
a = NaN || console.log("c");
|
|
||||||
a = undefined || 2 * condition;
|
|
||||||
a = null || condition + 3;
|
|
||||||
a = 2 * 3 - 6 || 4 - condition;
|
|
||||||
a = 10 == 7 || condition / 5;
|
|
||||||
a = !"string" || 6 % condition;
|
|
||||||
a = null || 7;
|
|
||||||
|
|
||||||
a = console.log(undefined && condition || null);
|
|
||||||
a = console.log(undefined || condition && null);
|
|
||||||
|
|
||||||
// don't compress these
|
|
||||||
|
|
||||||
a = condition || true;
|
|
||||||
a = console.log("a") || 2;
|
|
||||||
a = 4 - condition || "string";
|
|
||||||
a = 6 << condition || -4.5;
|
|
||||||
|
|
||||||
a = condition || false;
|
|
||||||
a = console.log("b") || NaN;
|
|
||||||
a = console.log("c") || 0;
|
|
||||||
a = 2 * condition || undefined;
|
|
||||||
a = condition + 3 || null;
|
|
||||||
|
|
||||||
}
|
|
||||||
expect: {
|
|
||||||
var a;
|
|
||||||
|
|
||||||
a = true;
|
|
||||||
a = 1;
|
|
||||||
a = 6;
|
|
||||||
a = true;
|
|
||||||
a = "string";
|
|
||||||
a = "5";
|
|
||||||
a = -4.5;
|
|
||||||
a = 6;
|
|
||||||
|
|
||||||
a = condition;
|
|
||||||
a = console.log("b");
|
|
||||||
a = console.log("c");
|
|
||||||
a = 2 * condition;
|
|
||||||
a = condition + 3;
|
|
||||||
a = 4 - condition;
|
|
||||||
a = condition / 5;
|
|
||||||
a = 6 % condition;
|
|
||||||
a = 7;
|
|
||||||
|
|
||||||
a = console.log(null);
|
|
||||||
a = console.log(condition && null);
|
|
||||||
|
|
||||||
a = condition || true;
|
|
||||||
a = console.log("a") || 2;
|
|
||||||
a = 4 - condition || "string";
|
|
||||||
a = 6 << condition || -4.5;
|
|
||||||
|
|
||||||
a = condition || false;
|
|
||||||
a = console.log("b") || NaN;
|
|
||||||
a = console.log("c") || 0;
|
|
||||||
a = 2 * condition || void 0;
|
|
||||||
a = condition + 3 || null;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
trivial_boolean_ternary_expressions : {
|
trivial_boolean_ternary_expressions : {
|
||||||
options = {
|
options = {
|
||||||
conditionals: true,
|
conditionals: true,
|
||||||
@@ -906,3 +746,26 @@ issue_1154: {
|
|||||||
function g6() { return g(), "number"; }
|
function g6() { return g(), "number"; }
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
no_evaluate: {
|
||||||
|
options = {
|
||||||
|
conditionals: true,
|
||||||
|
evaluate : false
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
function f(b) {
|
||||||
|
a = b ? !0 : !0;
|
||||||
|
a = b ? ~1 : ~1;
|
||||||
|
a = b ? -2 : -2;
|
||||||
|
a = b ? +3 : +3;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
function f(b) {
|
||||||
|
a = !0;
|
||||||
|
a = ~1;
|
||||||
|
a = -2;
|
||||||
|
a = +3;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|||||||
165
test/compress/const.js
Normal file
165
test/compress/const.js
Normal file
@@ -0,0 +1,165 @@
|
|||||||
|
issue_1191: {
|
||||||
|
options = {
|
||||||
|
evaluate : true,
|
||||||
|
booleans : true,
|
||||||
|
comparisons : true,
|
||||||
|
dead_code : true,
|
||||||
|
conditionals : true,
|
||||||
|
side_effects : true,
|
||||||
|
unused : true,
|
||||||
|
hoist_funs : true,
|
||||||
|
if_return : true,
|
||||||
|
join_vars : true,
|
||||||
|
sequences : false,
|
||||||
|
collapse_vars : false,
|
||||||
|
reduce_vars : true,
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
function foo(rot) {
|
||||||
|
const rotTol = 5;
|
||||||
|
if (rot < -rotTol || rot > rotTol)
|
||||||
|
bar();
|
||||||
|
baz();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
function foo(rot) {
|
||||||
|
(rot < -5 || rot > 5) && bar();
|
||||||
|
baz();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
issue_1194: {
|
||||||
|
options = {
|
||||||
|
evaluate : true,
|
||||||
|
booleans : true,
|
||||||
|
comparisons : true,
|
||||||
|
dead_code : true,
|
||||||
|
conditionals : true,
|
||||||
|
side_effects : true,
|
||||||
|
unused : true,
|
||||||
|
hoist_funs : true,
|
||||||
|
if_return : true,
|
||||||
|
join_vars : true,
|
||||||
|
sequences : false,
|
||||||
|
collapse_vars : false,
|
||||||
|
reduce_vars : true,
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
function f1() {const a = "X"; return a + a;}
|
||||||
|
function f2() {const aa = "X"; return aa + aa;}
|
||||||
|
function f3() {const aaa = "X"; return aaa + aaa;}
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
function f1(){return"XX"}
|
||||||
|
function f2(){return"XX"}
|
||||||
|
function f3(){return"XX"}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
issue_1396: {
|
||||||
|
options = {
|
||||||
|
evaluate : true,
|
||||||
|
booleans : true,
|
||||||
|
comparisons : true,
|
||||||
|
dead_code : true,
|
||||||
|
conditionals : true,
|
||||||
|
side_effects : true,
|
||||||
|
unused : true,
|
||||||
|
hoist_funs : true,
|
||||||
|
if_return : true,
|
||||||
|
join_vars : true,
|
||||||
|
sequences : false,
|
||||||
|
collapse_vars : false,
|
||||||
|
reduce_vars : true,
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
function foo(a) {
|
||||||
|
const VALUE = 1;
|
||||||
|
console.log(2 | VALUE);
|
||||||
|
console.log(VALUE + 1);
|
||||||
|
console.log(VALUE);
|
||||||
|
console.log(a & VALUE);
|
||||||
|
}
|
||||||
|
function bar() {
|
||||||
|
const s = "01234567890123456789";
|
||||||
|
console.log(s + s + s + s + s);
|
||||||
|
|
||||||
|
const CONSTANT = "abc";
|
||||||
|
console.log(CONSTANT + CONSTANT + CONSTANT + CONSTANT + CONSTANT);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
function foo(a) {
|
||||||
|
console.log(3);
|
||||||
|
console.log(2);
|
||||||
|
console.log(1);
|
||||||
|
console.log(1 & a);
|
||||||
|
}
|
||||||
|
function bar() {
|
||||||
|
const s = "01234567890123456789";
|
||||||
|
console.log(s + s + s + s + s);
|
||||||
|
|
||||||
|
console.log("abcabcabcabcabc");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
unused_regexp_literal: {
|
||||||
|
options = {
|
||||||
|
evaluate : true,
|
||||||
|
booleans : true,
|
||||||
|
comparisons : true,
|
||||||
|
dead_code : true,
|
||||||
|
conditionals : true,
|
||||||
|
side_effects : true,
|
||||||
|
unused : true,
|
||||||
|
hoist_funs : true,
|
||||||
|
if_return : true,
|
||||||
|
join_vars : true,
|
||||||
|
sequences : false,
|
||||||
|
collapse_vars : false,
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
function f(){ var a = /b/; }
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
function f(){}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
regexp_literal_not_const: {
|
||||||
|
options = {
|
||||||
|
evaluate : true,
|
||||||
|
booleans : true,
|
||||||
|
comparisons : true,
|
||||||
|
dead_code : true,
|
||||||
|
conditionals : true,
|
||||||
|
side_effects : true,
|
||||||
|
unused : true,
|
||||||
|
hoist_funs : true,
|
||||||
|
if_return : true,
|
||||||
|
join_vars : true,
|
||||||
|
sequences : false,
|
||||||
|
collapse_vars : false,
|
||||||
|
reduce_vars : true,
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
(function(){
|
||||||
|
var result;
|
||||||
|
const s = 'acdabcdeabbb';
|
||||||
|
const REGEXP_LITERAL = /ab*/g;
|
||||||
|
while (result = REGEXP_LITERAL.exec(s)) {
|
||||||
|
console.log(result[0]);
|
||||||
|
}
|
||||||
|
})();
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
(function() {
|
||||||
|
var result;
|
||||||
|
const REGEXP_LITERAL = /ab*/g;
|
||||||
|
while (result = REGEXP_LITERAL.exec("acdabcdeabbb")) console.log(result[0]);
|
||||||
|
})();
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,206 +1,208 @@
|
|||||||
dead_code_1: {
|
dead_code_1: {
|
||||||
options = {
|
options = {
|
||||||
dead_code: true
|
dead_code: true
|
||||||
};
|
};
|
||||||
input: {
|
input: {
|
||||||
function f() {
|
function f() {
|
||||||
a();
|
a();
|
||||||
b();
|
b();
|
||||||
x = 10;
|
x = 10;
|
||||||
return;
|
return;
|
||||||
if (x) {
|
if (x) {
|
||||||
y();
|
y();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
expect: {
|
expect: {
|
||||||
function f() {
|
function f() {
|
||||||
a();
|
a();
|
||||||
b();
|
b();
|
||||||
x = 10;
|
x = 10;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
dead_code_2_should_warn: {
|
dead_code_2_should_warn: {
|
||||||
options = {
|
options = {
|
||||||
dead_code: true
|
dead_code: true
|
||||||
};
|
};
|
||||||
input: {
|
input: {
|
||||||
function f() {
|
function f() {
|
||||||
g();
|
g();
|
||||||
x = 10;
|
x = 10;
|
||||||
throw "foo";
|
throw "foo";
|
||||||
// completely discarding the `if` would introduce some
|
// completely discarding the `if` would introduce some
|
||||||
// bugs. UglifyJS v1 doesn't deal with this issue; in v2
|
// bugs. UglifyJS v1 doesn't deal with this issue; in v2
|
||||||
// we copy any declarations to the upper scope.
|
// we copy any declarations to the upper scope.
|
||||||
if (x) {
|
if (x) {
|
||||||
y();
|
y();
|
||||||
var x;
|
var x;
|
||||||
function g(){};
|
function g(){};
|
||||||
// but nested declarations should not be kept.
|
// but nested declarations should not be kept.
|
||||||
(function(){
|
(function(){
|
||||||
var q;
|
var q;
|
||||||
function y(){};
|
function y(){};
|
||||||
})();
|
})();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
expect: {
|
expect: {
|
||||||
function f() {
|
function f() {
|
||||||
g();
|
g();
|
||||||
x = 10;
|
x = 10;
|
||||||
throw "foo";
|
throw "foo";
|
||||||
var x;
|
var x;
|
||||||
function g(){};
|
function g(){};
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
dead_code_constant_boolean_should_warn_more: {
|
dead_code_constant_boolean_should_warn_more: {
|
||||||
options = {
|
options = {
|
||||||
dead_code : true,
|
dead_code : true,
|
||||||
loops : true,
|
loops : true,
|
||||||
booleans : true,
|
booleans : true,
|
||||||
conditionals : true,
|
conditionals : true,
|
||||||
evaluate : true
|
evaluate : true
|
||||||
};
|
};
|
||||||
input: {
|
input: {
|
||||||
while (!((foo && bar) || (x + "0"))) {
|
while (!((foo && bar) || (x + "0"))) {
|
||||||
console.log("unreachable");
|
console.log("unreachable");
|
||||||
var foo;
|
var foo;
|
||||||
function bar() {}
|
function bar() {}
|
||||||
}
|
}
|
||||||
for (var x = 10, y; x && (y || x) && (!typeof x); ++x) {
|
for (var x = 10, y; x && (y || x) && (!typeof x); ++x) {
|
||||||
asdf();
|
asdf();
|
||||||
foo();
|
foo();
|
||||||
var moo;
|
var moo;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
expect: {
|
expect: {
|
||||||
var foo;
|
var foo;
|
||||||
function bar() {}
|
function bar() {}
|
||||||
// nothing for the while
|
// nothing for the while
|
||||||
// as for the for, it should keep:
|
// as for the for, it should keep:
|
||||||
var x = 10, y;
|
var x = 10, y;
|
||||||
var moo;
|
var moo;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
dead_code_const_declaration: {
|
dead_code_const_declaration: {
|
||||||
options = {
|
options = {
|
||||||
dead_code : true,
|
dead_code : true,
|
||||||
loops : true,
|
loops : true,
|
||||||
booleans : true,
|
booleans : true,
|
||||||
conditionals : true,
|
conditionals : true,
|
||||||
evaluate : true
|
evaluate : true,
|
||||||
};
|
reduce_vars : true,
|
||||||
input: {
|
};
|
||||||
var unused;
|
input: {
|
||||||
const CONST_FOO = false;
|
var unused;
|
||||||
if (CONST_FOO) {
|
const CONST_FOO = false;
|
||||||
console.log("unreachable");
|
if (CONST_FOO) {
|
||||||
var moo;
|
console.log("unreachable");
|
||||||
function bar() {}
|
var moo;
|
||||||
}
|
function bar() {}
|
||||||
}
|
}
|
||||||
expect: {
|
}
|
||||||
var unused;
|
expect: {
|
||||||
const CONST_FOO = !1;
|
var unused;
|
||||||
var moo;
|
const CONST_FOO = !1;
|
||||||
function bar() {}
|
var moo;
|
||||||
}
|
function bar() {}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
dead_code_const_annotation: {
|
|
||||||
options = {
|
dead_code_const_annotation: {
|
||||||
dead_code : true,
|
options = {
|
||||||
loops : true,
|
dead_code : true,
|
||||||
booleans : true,
|
loops : true,
|
||||||
conditionals : true,
|
booleans : true,
|
||||||
evaluate : true
|
conditionals : true,
|
||||||
};
|
evaluate : true,
|
||||||
input: {
|
reduce_vars : true,
|
||||||
var unused;
|
};
|
||||||
/** @const */ var CONST_FOO_ANN = false;
|
input: {
|
||||||
if (CONST_FOO_ANN) {
|
var unused;
|
||||||
console.log("unreachable");
|
/** @const */ var CONST_FOO_ANN = false;
|
||||||
var moo;
|
if (CONST_FOO_ANN) {
|
||||||
function bar() {}
|
console.log("unreachable");
|
||||||
}
|
var moo;
|
||||||
}
|
function bar() {}
|
||||||
expect: {
|
}
|
||||||
var unused;
|
}
|
||||||
var CONST_FOO_ANN = !1;
|
expect: {
|
||||||
var moo;
|
var unused;
|
||||||
function bar() {}
|
var CONST_FOO_ANN = !1;
|
||||||
}
|
var moo;
|
||||||
}
|
function bar() {}
|
||||||
|
}
|
||||||
dead_code_const_annotation_regex: {
|
}
|
||||||
options = {
|
|
||||||
dead_code : true,
|
dead_code_const_annotation_regex: {
|
||||||
loops : true,
|
options = {
|
||||||
booleans : true,
|
dead_code : true,
|
||||||
conditionals : true,
|
loops : true,
|
||||||
evaluate : true
|
booleans : true,
|
||||||
};
|
conditionals : true,
|
||||||
input: {
|
evaluate : true
|
||||||
var unused;
|
};
|
||||||
// @constraint this shouldn't be a constant
|
input: {
|
||||||
var CONST_FOO_ANN = false;
|
var unused;
|
||||||
if (CONST_FOO_ANN) {
|
// @constraint this shouldn't be a constant
|
||||||
console.log("reachable");
|
var CONST_FOO_ANN = false;
|
||||||
}
|
if (CONST_FOO_ANN) {
|
||||||
}
|
console.log("reachable");
|
||||||
expect: {
|
}
|
||||||
var unused;
|
}
|
||||||
var CONST_FOO_ANN = !1;
|
expect: {
|
||||||
CONST_FOO_ANN && console.log('reachable');
|
var unused;
|
||||||
}
|
var CONST_FOO_ANN = !1;
|
||||||
}
|
CONST_FOO_ANN && console.log('reachable');
|
||||||
|
}
|
||||||
dead_code_const_annotation_complex_scope: {
|
}
|
||||||
options = {
|
|
||||||
dead_code : true,
|
dead_code_const_annotation_complex_scope: {
|
||||||
loops : true,
|
options = {
|
||||||
booleans : true,
|
dead_code : true,
|
||||||
conditionals : true,
|
loops : true,
|
||||||
evaluate : true
|
booleans : true,
|
||||||
};
|
conditionals : true,
|
||||||
input: {
|
evaluate : true,
|
||||||
var unused_var;
|
reduce_vars : true,
|
||||||
/** @const */ var test = 'test';
|
};
|
||||||
// @const
|
input: {
|
||||||
var CONST_FOO_ANN = false;
|
var unused_var;
|
||||||
var unused_var_2;
|
/** @const */ var test = 'test';
|
||||||
if (CONST_FOO_ANN) {
|
// @const
|
||||||
console.log("unreachable");
|
var CONST_FOO_ANN = false;
|
||||||
var moo;
|
var unused_var_2;
|
||||||
function bar() {}
|
if (CONST_FOO_ANN) {
|
||||||
}
|
console.log("unreachable");
|
||||||
if (test === 'test') {
|
var moo;
|
||||||
var beef = 'good';
|
function bar() {}
|
||||||
/** @const */ var meat = 'beef';
|
}
|
||||||
var pork = 'bad';
|
if (test === 'test') {
|
||||||
if (meat === 'pork') {
|
var beef = 'good';
|
||||||
console.log('also unreachable');
|
/** @const */ var meat = 'beef';
|
||||||
} else if (pork === 'good') {
|
var pork = 'bad';
|
||||||
console.log('reached, not const');
|
if (meat === 'pork') {
|
||||||
}
|
console.log('also unreachable');
|
||||||
}
|
} else if (pork === 'good') {
|
||||||
}
|
console.log('reached, not const');
|
||||||
expect: {
|
}
|
||||||
var unused_var;
|
}
|
||||||
var test = 'test';
|
}
|
||||||
var CONST_FOO_ANN = !1;
|
expect: {
|
||||||
var unused_var_2;
|
var unused_var;
|
||||||
var moo;
|
var test = 'test';
|
||||||
function bar() {}
|
var CONST_FOO_ANN = !1;
|
||||||
var beef = 'good';
|
var unused_var_2;
|
||||||
var meat = 'beef';
|
var moo;
|
||||||
var pork = 'bad';
|
function bar() {}
|
||||||
'good' === pork && console.log('reached, not const');
|
var beef = 'good';
|
||||||
}
|
var meat = 'beef';
|
||||||
}
|
var pork = 'bad';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|||||||
@@ -1,24 +1,24 @@
|
|||||||
drop_console_1: {
|
drop_console_1: {
|
||||||
options = {};
|
options = {};
|
||||||
input: {
|
input: {
|
||||||
console.log('foo');
|
console.log('foo');
|
||||||
console.log.apply(console, arguments);
|
console.log.apply(console, arguments);
|
||||||
}
|
}
|
||||||
expect: {
|
expect: {
|
||||||
console.log('foo');
|
console.log('foo');
|
||||||
console.log.apply(console, arguments);
|
console.log.apply(console, arguments);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
drop_console_1: {
|
drop_console_2: {
|
||||||
options = { drop_console: true };
|
options = { drop_console: true };
|
||||||
input: {
|
input: {
|
||||||
console.log('foo');
|
console.log('foo');
|
||||||
console.log.apply(console, arguments);
|
console.log.apply(console, arguments);
|
||||||
}
|
}
|
||||||
expect: {
|
expect: {
|
||||||
// with regular compression these will be stripped out as well
|
// with regular compression these will be stripped out as well
|
||||||
void 0;
|
void 0;
|
||||||
void 0;
|
void 0;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -177,3 +177,505 @@ keep_fnames: {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
drop_assign: {
|
||||||
|
options = { unused: true };
|
||||||
|
input: {
|
||||||
|
function f1() {
|
||||||
|
var a;
|
||||||
|
a = 1;
|
||||||
|
}
|
||||||
|
function f2() {
|
||||||
|
var a = 1;
|
||||||
|
a = 2;
|
||||||
|
}
|
||||||
|
function f3(a) {
|
||||||
|
a = 1;
|
||||||
|
}
|
||||||
|
function f4() {
|
||||||
|
var a;
|
||||||
|
return a = 1;
|
||||||
|
}
|
||||||
|
function f5() {
|
||||||
|
var a;
|
||||||
|
return function() {
|
||||||
|
a = 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
function f1() {
|
||||||
|
1;
|
||||||
|
}
|
||||||
|
function f2() {
|
||||||
|
2;
|
||||||
|
}
|
||||||
|
function f3(a) {
|
||||||
|
1;
|
||||||
|
}
|
||||||
|
function f4() {
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
function f5() {
|
||||||
|
var a;
|
||||||
|
return function() {
|
||||||
|
a = 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
keep_assign: {
|
||||||
|
options = { unused: "keep_assign" };
|
||||||
|
input: {
|
||||||
|
function f1() {
|
||||||
|
var a;
|
||||||
|
a = 1;
|
||||||
|
}
|
||||||
|
function f2() {
|
||||||
|
var a = 1;
|
||||||
|
a = 2;
|
||||||
|
}
|
||||||
|
function f3(a) {
|
||||||
|
a = 1;
|
||||||
|
}
|
||||||
|
function f4() {
|
||||||
|
var a;
|
||||||
|
return a = 1;
|
||||||
|
}
|
||||||
|
function f5() {
|
||||||
|
var a;
|
||||||
|
return function() {
|
||||||
|
a = 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
function f1() {
|
||||||
|
var a;
|
||||||
|
a = 1;
|
||||||
|
}
|
||||||
|
function f2() {
|
||||||
|
var a = 1;
|
||||||
|
a = 2;
|
||||||
|
}
|
||||||
|
function f3(a) {
|
||||||
|
a = 1;
|
||||||
|
}
|
||||||
|
function f4() {
|
||||||
|
var a;
|
||||||
|
return a = 1;
|
||||||
|
}
|
||||||
|
function f5() {
|
||||||
|
var a;
|
||||||
|
return function() {
|
||||||
|
a = 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
drop_toplevel_funcs: {
|
||||||
|
options = { toplevel: "funcs", unused: true };
|
||||||
|
input: {
|
||||||
|
var a, b = 1, c = g;
|
||||||
|
function f(d) {
|
||||||
|
return function() {
|
||||||
|
c = 2;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
a = 2;
|
||||||
|
function g() {}
|
||||||
|
function h() {}
|
||||||
|
console.log(b = 3);
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
var a, b = 1, c = g;
|
||||||
|
a = 2;
|
||||||
|
function g() {}
|
||||||
|
console.log(b = 3);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
drop_toplevel_vars: {
|
||||||
|
options = { toplevel: "vars", unused: true };
|
||||||
|
input: {
|
||||||
|
var a, b = 1, c = g;
|
||||||
|
function f(d) {
|
||||||
|
return function() {
|
||||||
|
c = 2;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
a = 2;
|
||||||
|
function g() {}
|
||||||
|
function h() {}
|
||||||
|
console.log(b = 3);
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
var c = g;
|
||||||
|
function f(d) {
|
||||||
|
return function() {
|
||||||
|
c = 2;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
2;
|
||||||
|
function g() {}
|
||||||
|
function h() {}
|
||||||
|
console.log(3);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
drop_toplevel_vars_fargs: {
|
||||||
|
options = { keep_fargs: false, toplevel: "vars", unused: true };
|
||||||
|
input: {
|
||||||
|
var a, b = 1, c = g;
|
||||||
|
function f(d) {
|
||||||
|
return function() {
|
||||||
|
c = 2;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
a = 2;
|
||||||
|
function g() {}
|
||||||
|
function h() {}
|
||||||
|
console.log(b = 3);
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
var c = g;
|
||||||
|
function f() {
|
||||||
|
return function() {
|
||||||
|
c = 2;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
2;
|
||||||
|
function g() {}
|
||||||
|
function h() {}
|
||||||
|
console.log(3);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
drop_toplevel_all: {
|
||||||
|
options = { toplevel: true, unused: true };
|
||||||
|
input: {
|
||||||
|
var a, b = 1, c = g;
|
||||||
|
function f(d) {
|
||||||
|
return function() {
|
||||||
|
c = 2;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
a = 2;
|
||||||
|
function g() {}
|
||||||
|
function h() {}
|
||||||
|
console.log(b = 3);
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
2;
|
||||||
|
console.log(3);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
drop_toplevel_retain: {
|
||||||
|
options = { top_retain: "f,a,o", unused: true };
|
||||||
|
input: {
|
||||||
|
var a, b = 1, c = g;
|
||||||
|
function f(d) {
|
||||||
|
return function() {
|
||||||
|
c = 2;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
a = 2;
|
||||||
|
function g() {}
|
||||||
|
function h() {}
|
||||||
|
console.log(b = 3);
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
var a, c = g;
|
||||||
|
function f(d) {
|
||||||
|
return function() {
|
||||||
|
c = 2;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
a = 2;
|
||||||
|
function g() {}
|
||||||
|
console.log(3);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
drop_toplevel_retain_array: {
|
||||||
|
options = { top_retain: [ "f", "a", "o" ], unused: true };
|
||||||
|
input: {
|
||||||
|
var a, b = 1, c = g;
|
||||||
|
function f(d) {
|
||||||
|
return function() {
|
||||||
|
c = 2;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
a = 2;
|
||||||
|
function g() {}
|
||||||
|
function h() {}
|
||||||
|
console.log(b = 3);
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
var a, c = g;
|
||||||
|
function f(d) {
|
||||||
|
return function() {
|
||||||
|
c = 2;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
a = 2;
|
||||||
|
function g() {}
|
||||||
|
console.log(3);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
drop_toplevel_retain_regex: {
|
||||||
|
options = { top_retain: /^[fao]$/, unused: true };
|
||||||
|
input: {
|
||||||
|
var a, b = 1, c = g;
|
||||||
|
function f(d) {
|
||||||
|
return function() {
|
||||||
|
c = 2;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
a = 2;
|
||||||
|
function g() {}
|
||||||
|
function h() {}
|
||||||
|
console.log(b = 3);
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
var a, c = g;
|
||||||
|
function f(d) {
|
||||||
|
return function() {
|
||||||
|
c = 2;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
a = 2;
|
||||||
|
function g() {}
|
||||||
|
console.log(3);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
drop_toplevel_all_retain: {
|
||||||
|
options = { toplevel: true, top_retain: "f,a,o", unused: true };
|
||||||
|
input: {
|
||||||
|
var a, b = 1, c = g;
|
||||||
|
function f(d) {
|
||||||
|
return function() {
|
||||||
|
c = 2;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
a = 2;
|
||||||
|
function g() {}
|
||||||
|
function h() {}
|
||||||
|
console.log(b = 3);
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
var a, c = g;
|
||||||
|
function f(d) {
|
||||||
|
return function() {
|
||||||
|
c = 2;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
a = 2;
|
||||||
|
function g() {}
|
||||||
|
console.log(3);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
drop_toplevel_funcs_retain: {
|
||||||
|
options = { toplevel: "funcs", top_retain: "f,a,o", unused: true };
|
||||||
|
input: {
|
||||||
|
var a, b = 1, c = g;
|
||||||
|
function f(d) {
|
||||||
|
return function() {
|
||||||
|
c = 2;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
a = 2;
|
||||||
|
function g() {}
|
||||||
|
function h() {}
|
||||||
|
console.log(b = 3);
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
var a, b = 1, c = g;
|
||||||
|
function f(d) {
|
||||||
|
return function() {
|
||||||
|
c = 2;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
a = 2;
|
||||||
|
function g() {}
|
||||||
|
console.log(b = 3);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
drop_toplevel_vars_retain: {
|
||||||
|
options = { toplevel: "vars", top_retain: "f,a,o", unused: true };
|
||||||
|
input: {
|
||||||
|
var a, b = 1, c = g;
|
||||||
|
function f(d) {
|
||||||
|
return function() {
|
||||||
|
c = 2;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
a = 2;
|
||||||
|
function g() {}
|
||||||
|
function h() {}
|
||||||
|
console.log(b = 3);
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
var a, c = g;
|
||||||
|
function f(d) {
|
||||||
|
return function() {
|
||||||
|
c = 2;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
a = 2;
|
||||||
|
function g() {}
|
||||||
|
function h() {}
|
||||||
|
console.log(3);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
drop_toplevel_keep_assign: {
|
||||||
|
options = { toplevel: true, unused: "keep_assign" };
|
||||||
|
input: {
|
||||||
|
var a, b = 1, c = g;
|
||||||
|
function f(d) {
|
||||||
|
return function() {
|
||||||
|
c = 2;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
a = 2;
|
||||||
|
function g() {}
|
||||||
|
function h() {}
|
||||||
|
console.log(b = 3);
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
var a, b = 1;
|
||||||
|
a = 2;
|
||||||
|
console.log(b = 3);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
drop_fargs: {
|
||||||
|
options = {
|
||||||
|
keep_fargs: false,
|
||||||
|
unused: true,
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
function f(a) {
|
||||||
|
var b = a;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
function f() {}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
drop_fnames: {
|
||||||
|
options = {
|
||||||
|
keep_fnames: false,
|
||||||
|
unused: true,
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
function f() {
|
||||||
|
return function g() {
|
||||||
|
var a = g;
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
function f() {
|
||||||
|
return function() {};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
global_var: {
|
||||||
|
options = {
|
||||||
|
side_effects: true,
|
||||||
|
unused: true,
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
var a;
|
||||||
|
function foo(b) {
|
||||||
|
a;
|
||||||
|
b;
|
||||||
|
c;
|
||||||
|
typeof c === "undefined";
|
||||||
|
c + b + a;
|
||||||
|
b && b.ar();
|
||||||
|
return b;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
var a;
|
||||||
|
function foo(b) {
|
||||||
|
c;
|
||||||
|
c;
|
||||||
|
b && b.ar();
|
||||||
|
return b;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
iife: {
|
||||||
|
options = {
|
||||||
|
side_effects: true,
|
||||||
|
unused: true,
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
function f() {
|
||||||
|
var a;
|
||||||
|
~function() {}(b);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
function f() {
|
||||||
|
~function() {}(b);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
drop_value: {
|
||||||
|
options = {
|
||||||
|
side_effects: true,
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
(1, [2, foo()], 3, {a:1, b:bar()});
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
foo(), bar();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const_assign: {
|
||||||
|
options = {
|
||||||
|
evaluate: true,
|
||||||
|
reduce_vars: true,
|
||||||
|
unused: true,
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
function f() {
|
||||||
|
const b = 2;
|
||||||
|
return 1 + b;
|
||||||
|
}
|
||||||
|
|
||||||
|
function g() {
|
||||||
|
const b = 2;
|
||||||
|
b = 3;
|
||||||
|
return 1 + b;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
function f() {
|
||||||
|
return 3;
|
||||||
|
}
|
||||||
|
|
||||||
|
function g() {
|
||||||
|
const b = 2;
|
||||||
|
b = 3;
|
||||||
|
return 1 + b;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|||||||
@@ -1,3 +1,187 @@
|
|||||||
|
and: {
|
||||||
|
options = {
|
||||||
|
evaluate: true
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
var a;
|
||||||
|
// compress these
|
||||||
|
|
||||||
|
a = true && condition;
|
||||||
|
a = 1 && console.log("a");
|
||||||
|
a = 2 * 3 && 2 * condition;
|
||||||
|
a = 5 == 5 && condition + 3;
|
||||||
|
a = "string" && 4 - condition;
|
||||||
|
a = 5 + "" && condition / 5;
|
||||||
|
a = -4.5 && 6 << condition;
|
||||||
|
a = 6 && 7;
|
||||||
|
|
||||||
|
a = false && condition;
|
||||||
|
a = NaN && console.log("b");
|
||||||
|
a = 0 && console.log("c");
|
||||||
|
a = undefined && 2 * condition;
|
||||||
|
a = null && condition + 3;
|
||||||
|
a = 2 * 3 - 6 && 4 - condition;
|
||||||
|
a = 10 == 7 && condition / 5;
|
||||||
|
a = !"string" && 6 % condition;
|
||||||
|
a = 0 && 7;
|
||||||
|
|
||||||
|
// don't compress these
|
||||||
|
|
||||||
|
a = condition && true;
|
||||||
|
a = console.log("a") && 2;
|
||||||
|
a = 4 - condition && "string";
|
||||||
|
a = 6 << condition && -4.5;
|
||||||
|
|
||||||
|
a = condition && false;
|
||||||
|
a = console.log("b") && NaN;
|
||||||
|
a = console.log("c") && 0;
|
||||||
|
a = 2 * condition && undefined;
|
||||||
|
a = condition + 3 && null;
|
||||||
|
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
var a;
|
||||||
|
|
||||||
|
a = condition;
|
||||||
|
a = console.log("a");
|
||||||
|
a = 2 * condition;
|
||||||
|
a = condition + 3;
|
||||||
|
a = 4 - condition;
|
||||||
|
a = condition / 5;
|
||||||
|
a = 6 << condition;
|
||||||
|
a = 7;
|
||||||
|
|
||||||
|
a = false;
|
||||||
|
a = NaN;
|
||||||
|
a = 0;
|
||||||
|
a = void 0;
|
||||||
|
a = null;
|
||||||
|
a = 0;
|
||||||
|
a = false;
|
||||||
|
a = false;
|
||||||
|
a = 0;
|
||||||
|
|
||||||
|
a = condition && true;
|
||||||
|
a = console.log("a") && 2;
|
||||||
|
a = 4 - condition && "string";
|
||||||
|
a = 6 << condition && -4.5;
|
||||||
|
|
||||||
|
a = condition && false;
|
||||||
|
a = console.log("b") && NaN;
|
||||||
|
a = console.log("c") && 0;
|
||||||
|
a = 2 * condition && void 0;
|
||||||
|
a = condition + 3 && null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
or: {
|
||||||
|
options = {
|
||||||
|
evaluate: true
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
var a;
|
||||||
|
// compress these
|
||||||
|
|
||||||
|
a = true || condition;
|
||||||
|
a = 1 || console.log("a");
|
||||||
|
a = 2 * 3 || 2 * condition;
|
||||||
|
a = 5 == 5 || condition + 3;
|
||||||
|
a = "string" || 4 - condition;
|
||||||
|
a = 5 + "" || condition / 5;
|
||||||
|
a = -4.5 || 6 << condition;
|
||||||
|
a = 6 || 7;
|
||||||
|
|
||||||
|
a = false || condition;
|
||||||
|
a = 0 || console.log("b");
|
||||||
|
a = NaN || console.log("c");
|
||||||
|
a = undefined || 2 * condition;
|
||||||
|
a = null || condition + 3;
|
||||||
|
a = 2 * 3 - 6 || 4 - condition;
|
||||||
|
a = 10 == 7 || condition / 5;
|
||||||
|
a = !"string" || 6 % condition;
|
||||||
|
a = null || 7;
|
||||||
|
|
||||||
|
a = console.log(undefined && condition || null);
|
||||||
|
a = console.log(undefined || condition && null);
|
||||||
|
|
||||||
|
// don't compress these
|
||||||
|
|
||||||
|
a = condition || true;
|
||||||
|
a = console.log("a") || 2;
|
||||||
|
a = 4 - condition || "string";
|
||||||
|
a = 6 << condition || -4.5;
|
||||||
|
|
||||||
|
a = condition || false;
|
||||||
|
a = console.log("b") || NaN;
|
||||||
|
a = console.log("c") || 0;
|
||||||
|
a = 2 * condition || undefined;
|
||||||
|
a = condition + 3 || null;
|
||||||
|
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
var a;
|
||||||
|
|
||||||
|
a = true;
|
||||||
|
a = 1;
|
||||||
|
a = 6;
|
||||||
|
a = true;
|
||||||
|
a = "string";
|
||||||
|
a = "5";
|
||||||
|
a = -4.5;
|
||||||
|
a = 6;
|
||||||
|
|
||||||
|
a = condition;
|
||||||
|
a = console.log("b");
|
||||||
|
a = console.log("c");
|
||||||
|
a = 2 * condition;
|
||||||
|
a = condition + 3;
|
||||||
|
a = 4 - condition;
|
||||||
|
a = condition / 5;
|
||||||
|
a = 6 % condition;
|
||||||
|
a = 7;
|
||||||
|
|
||||||
|
a = console.log(null);
|
||||||
|
a = console.log(condition && null);
|
||||||
|
|
||||||
|
a = condition || true;
|
||||||
|
a = console.log("a") || 2;
|
||||||
|
a = 4 - condition || "string";
|
||||||
|
a = 6 << condition || -4.5;
|
||||||
|
|
||||||
|
a = condition || false;
|
||||||
|
a = console.log("b") || NaN;
|
||||||
|
a = console.log("c") || 0;
|
||||||
|
a = 2 * condition || void 0;
|
||||||
|
a = condition + 3 || null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
unary_prefix: {
|
||||||
|
options = {
|
||||||
|
evaluate: true
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
a = !0 && b;
|
||||||
|
a = !0 || b;
|
||||||
|
a = ~1 && b;
|
||||||
|
a = ~1 || b;
|
||||||
|
a = -2 && b;
|
||||||
|
a = -2 || b;
|
||||||
|
a = +3 && b;
|
||||||
|
a = +3 || b;
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
a = b;
|
||||||
|
a = !0;
|
||||||
|
a = b;
|
||||||
|
a = -2;
|
||||||
|
a = b;
|
||||||
|
a = -2;
|
||||||
|
a = b;
|
||||||
|
a = 3;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
negative_zero: {
|
negative_zero: {
|
||||||
options = { evaluate: true }
|
options = { evaluate: true }
|
||||||
input: {
|
input: {
|
||||||
@@ -37,3 +221,428 @@ positive_zero: {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
unsafe_constant: {
|
||||||
|
options = {
|
||||||
|
evaluate : true,
|
||||||
|
unsafe : true
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
console.log(
|
||||||
|
true.a,
|
||||||
|
false.a,
|
||||||
|
null.a,
|
||||||
|
undefined.a
|
||||||
|
);
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
console.log(
|
||||||
|
true.a,
|
||||||
|
false.a,
|
||||||
|
null.a,
|
||||||
|
(void 0).a
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
unsafe_object: {
|
||||||
|
options = {
|
||||||
|
evaluate : true,
|
||||||
|
unsafe : true
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
console.log(
|
||||||
|
({a:1}) + 1,
|
||||||
|
({a:1}).a + 1,
|
||||||
|
({a:1}).b + 1,
|
||||||
|
({a:1}).a.b + 1
|
||||||
|
);
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
console.log(
|
||||||
|
({a:1}) + 1,
|
||||||
|
2,
|
||||||
|
({a:1}).b + 1,
|
||||||
|
1..b + 1
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
unsafe_object_nested: {
|
||||||
|
options = {
|
||||||
|
evaluate : true,
|
||||||
|
unsafe : true
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
console.log(
|
||||||
|
({a:{b:1}}) + 1,
|
||||||
|
({a:{b:1}}).a + 1,
|
||||||
|
({a:{b:1}}).b + 1,
|
||||||
|
({a:{b:1}}).a.b + 1
|
||||||
|
);
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
console.log(
|
||||||
|
({a:{b:1}}) + 1,
|
||||||
|
({a:{b:1}}).a + 1,
|
||||||
|
({a:{b:1}}).b + 1,
|
||||||
|
2
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
unsafe_object_complex: {
|
||||||
|
options = {
|
||||||
|
evaluate : true,
|
||||||
|
unsafe : true
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
console.log(
|
||||||
|
({a:{b:1},b:1}) + 1,
|
||||||
|
({a:{b:1},b:1}).a + 1,
|
||||||
|
({a:{b:1},b:1}).b + 1,
|
||||||
|
({a:{b:1},b:1}).a.b + 1
|
||||||
|
);
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
console.log(
|
||||||
|
({a:{b:1},b:1}) + 1,
|
||||||
|
({a:{b:1},b:1}).a + 1,
|
||||||
|
2,
|
||||||
|
2
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
unsafe_object_repeated: {
|
||||||
|
options = {
|
||||||
|
evaluate : true,
|
||||||
|
unsafe : true
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
console.log(
|
||||||
|
({a:{b:1},a:1}) + 1,
|
||||||
|
({a:{b:1},a:1}).a + 1,
|
||||||
|
({a:{b:1},a:1}).b + 1,
|
||||||
|
({a:{b:1},a:1}).a.b + 1
|
||||||
|
);
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
console.log(
|
||||||
|
({a:{b:1},a:1}) + 1,
|
||||||
|
2,
|
||||||
|
({a:{b:1},a:1}).b + 1,
|
||||||
|
1..b + 1
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
unsafe_function: {
|
||||||
|
options = {
|
||||||
|
evaluate : true,
|
||||||
|
unsafe : true
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
console.log(
|
||||||
|
({a:{b:1},b:function(){}}) + 1,
|
||||||
|
({a:{b:1},b:function(){}}).a + 1,
|
||||||
|
({a:{b:1},b:function(){}}).b + 1,
|
||||||
|
({a:{b:1},b:function(){}}).a.b + 1
|
||||||
|
);
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
console.log(
|
||||||
|
({a:{b:1},b:function(){}}) + 1,
|
||||||
|
({a:{b:1},b:function(){}}).a + 1,
|
||||||
|
({a:{b:1},b:function(){}}).b + 1,
|
||||||
|
({a:{b:1},b:function(){}}).a.b + 1
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
unsafe_integer_key: {
|
||||||
|
options = {
|
||||||
|
evaluate : true,
|
||||||
|
unsafe : true
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
console.log(
|
||||||
|
({0:1}) + 1,
|
||||||
|
({0:1})[0] + 1,
|
||||||
|
({0:1})["0"] + 1,
|
||||||
|
({0:1})[1] + 1,
|
||||||
|
({0:1})[0][1] + 1,
|
||||||
|
({0:1})[0]["1"] + 1
|
||||||
|
);
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
console.log(
|
||||||
|
({0:1}) + 1,
|
||||||
|
2,
|
||||||
|
2,
|
||||||
|
({0:1})[1] + 1,
|
||||||
|
1[1] + 1,
|
||||||
|
1["1"] + 1
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
unsafe_integer_key_complex: {
|
||||||
|
options = {
|
||||||
|
evaluate : true,
|
||||||
|
unsafe : true
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
console.log(
|
||||||
|
({0:{1:1},1:1}) + 1,
|
||||||
|
({0:{1:1},1:1})[0] + 1,
|
||||||
|
({0:{1:1},1:1})["0"] + 1,
|
||||||
|
({0:{1:1},1:1})[1] + 1,
|
||||||
|
({0:{1:1},1:1})[0][1] + 1,
|
||||||
|
({0:{1:1},1:1})[0]["1"] + 1
|
||||||
|
);
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
console.log(
|
||||||
|
({0:{1:1},1:1}) + 1,
|
||||||
|
"[object Object]1",
|
||||||
|
"[object Object]1",
|
||||||
|
2,
|
||||||
|
2,
|
||||||
|
2
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
unsafe_float_key: {
|
||||||
|
options = {
|
||||||
|
evaluate : true,
|
||||||
|
unsafe : true
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
console.log(
|
||||||
|
({2.72:1}) + 1,
|
||||||
|
({2.72:1})[2.72] + 1,
|
||||||
|
({2.72:1})["2.72"] + 1,
|
||||||
|
({2.72:1})[3.14] + 1,
|
||||||
|
({2.72:1})[2.72][3.14] + 1,
|
||||||
|
({2.72:1})[2.72]["3.14"] + 1
|
||||||
|
);
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
console.log(
|
||||||
|
({2.72:1}) + 1,
|
||||||
|
2,
|
||||||
|
2,
|
||||||
|
({2.72:1})[3.14] + 1,
|
||||||
|
1[3.14] + 1,
|
||||||
|
1["3.14"] + 1
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
unsafe_float_key_complex: {
|
||||||
|
options = {
|
||||||
|
evaluate : true,
|
||||||
|
unsafe : true
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
console.log(
|
||||||
|
({2.72:{3.14:1},3.14:1}) + 1,
|
||||||
|
({2.72:{3.14:1},3.14:1})[2.72] + 1,
|
||||||
|
({2.72:{3.14:1},3.14:1})["2.72"] + 1,
|
||||||
|
({2.72:{3.14:1},3.14:1})[3.14] + 1,
|
||||||
|
({2.72:{3.14:1},3.14:1})[2.72][3.14] + 1,
|
||||||
|
({2.72:{3.14:1},3.14:1})[2.72]["3.14"] + 1
|
||||||
|
);
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
console.log(
|
||||||
|
"[object Object]1",
|
||||||
|
"[object Object]1",
|
||||||
|
"[object Object]1",
|
||||||
|
2,
|
||||||
|
2,
|
||||||
|
2
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
unsafe_array: {
|
||||||
|
options = {
|
||||||
|
evaluate : true,
|
||||||
|
unsafe : true
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
console.log(
|
||||||
|
[1, , 3][1],
|
||||||
|
[1, 2, 3, a] + 1,
|
||||||
|
[1, 2, 3, 4] + 1,
|
||||||
|
[1, 2, 3, a][0] + 1,
|
||||||
|
[1, 2, 3, 4][0] + 1,
|
||||||
|
[1, 2, 3, 4][6 - 5] + 1,
|
||||||
|
[1, , 3, 4][6 - 5] + 1,
|
||||||
|
[[1, 2], [3, 4]][0] + 1,
|
||||||
|
[[1, 2], [3, 4]][6 - 5][1] + 1,
|
||||||
|
[[1, 2], , [3, 4]][6 - 5][1] + 1
|
||||||
|
);
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
console.log(
|
||||||
|
void 0,
|
||||||
|
[1, 2, 3, a] + 1,
|
||||||
|
"1,2,3,41",
|
||||||
|
[1, 2, 3, a][0] + 1,
|
||||||
|
2,
|
||||||
|
3,
|
||||||
|
NaN,
|
||||||
|
"1,21",
|
||||||
|
5,
|
||||||
|
(void 0)[1] + 1
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
unsafe_string: {
|
||||||
|
options = {
|
||||||
|
evaluate : true,
|
||||||
|
unsafe : true
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
console.log(
|
||||||
|
"1234" + 1,
|
||||||
|
"1234"[0] + 1,
|
||||||
|
"1234"[6 - 5] + 1,
|
||||||
|
("12" + "34")[0] + 1,
|
||||||
|
("12" + "34")[6 - 5] + 1,
|
||||||
|
[1, 2, 3, 4].join("")[0] + 1
|
||||||
|
);
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
console.log(
|
||||||
|
"12341",
|
||||||
|
"11",
|
||||||
|
"21",
|
||||||
|
"11",
|
||||||
|
"21",
|
||||||
|
"11"
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
unsafe_array_bad_index: {
|
||||||
|
options = {
|
||||||
|
evaluate : true,
|
||||||
|
unsafe : true
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
console.log(
|
||||||
|
[1, 2, 3, 4].a + 1,
|
||||||
|
[1, 2, 3, 4]["a"] + 1,
|
||||||
|
[1, 2, 3, 4][3.14] + 1
|
||||||
|
);
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
console.log(
|
||||||
|
[1, 2, 3, 4].a + 1,
|
||||||
|
[1, 2, 3, 4]["a"] + 1,
|
||||||
|
[1, 2, 3, 4][3.14] + 1
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
unsafe_string_bad_index: {
|
||||||
|
options = {
|
||||||
|
evaluate : true,
|
||||||
|
unsafe : true
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
console.log(
|
||||||
|
"1234".a + 1,
|
||||||
|
"1234"["a"] + 1,
|
||||||
|
"1234"[3.14] + 1
|
||||||
|
);
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
console.log(
|
||||||
|
"1234".a + 1,
|
||||||
|
"1234"["a"] + 1,
|
||||||
|
"1234"[3.14] + 1
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
unsafe_prototype_function: {
|
||||||
|
options = {
|
||||||
|
evaluate : true,
|
||||||
|
unsafe : 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();
|
||||||
|
}
|
||||||
|
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 = ({valueOf: 0}).valueOf();
|
||||||
|
var h = "" + ({toString: 0});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
call_args: {
|
||||||
|
options = {
|
||||||
|
evaluate: true,
|
||||||
|
reduce_vars: true,
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
const a = 1;
|
||||||
|
console.log(a);
|
||||||
|
+function(a) {
|
||||||
|
return a;
|
||||||
|
}(a);
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
const a = 1;
|
||||||
|
console.log(1);
|
||||||
|
+function(a) {
|
||||||
|
return 1;
|
||||||
|
}(1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
in_boolean_context: {
|
||||||
|
options = {
|
||||||
|
booleans: true,
|
||||||
|
evaluate: true,
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
!42;
|
||||||
|
!"foo";
|
||||||
|
![1, 2];
|
||||||
|
!/foo/;
|
||||||
|
!b(42);
|
||||||
|
!b("foo");
|
||||||
|
!b([1, 2]);
|
||||||
|
!b(/foo/);
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
!1;
|
||||||
|
!1;
|
||||||
|
!1;
|
||||||
|
!1;
|
||||||
|
!b(42);
|
||||||
|
!b("foo");
|
||||||
|
!b([1, 2]);
|
||||||
|
!b(/foo/);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|||||||
8
test/compress/functions.js
Normal file
8
test/compress/functions.js
Normal file
@@ -0,0 +1,8 @@
|
|||||||
|
non_ascii_function_identifier_name: {
|
||||||
|
input: {
|
||||||
|
function fooλ(δλ) {}
|
||||||
|
function λ(δλ) {}
|
||||||
|
(function λ(δλ) {})()
|
||||||
|
}
|
||||||
|
expect_exact: "function fooλ(δλ){}function λ(δλ){}(function λ(δλ){})();"
|
||||||
|
}
|
||||||
147
test/compress/global_defs.js
Normal file
147
test/compress/global_defs.js
Normal file
@@ -0,0 +1,147 @@
|
|||||||
|
must_replace: {
|
||||||
|
options = {
|
||||||
|
global_defs: {
|
||||||
|
D: "foo bar",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
console.log(D);
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
console.log("foo bar");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
keyword: {
|
||||||
|
options = {
|
||||||
|
global_defs: {
|
||||||
|
undefined: 0,
|
||||||
|
NaN: 1,
|
||||||
|
Infinity: 2,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
console.log(undefined, NaN, Infinity);
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
console.log(0, 1, 2);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
object: {
|
||||||
|
options = {
|
||||||
|
evaluate: true,
|
||||||
|
global_defs: {
|
||||||
|
CONFIG: {
|
||||||
|
DEBUG: [ 0 ],
|
||||||
|
VALUE: 42,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
unsafe: true,
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
function f(CONFIG) {
|
||||||
|
// CONFIG not global - do not replace
|
||||||
|
return CONFIG.VALUE;
|
||||||
|
}
|
||||||
|
function g() {
|
||||||
|
var CONFIG = { VALUE: 1 };
|
||||||
|
// CONFIG not global - do not replace
|
||||||
|
return CONFIG.VALUE;
|
||||||
|
}
|
||||||
|
function h() {
|
||||||
|
return CONFIG.VALUE;
|
||||||
|
}
|
||||||
|
if (CONFIG.DEBUG[0])
|
||||||
|
console.debug("foo");
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
function f(CONFIG) {
|
||||||
|
return CONFIG.VALUE;
|
||||||
|
}
|
||||||
|
function g() {
|
||||||
|
var CONFIG = { VALUE: 1 };
|
||||||
|
return CONFIG.VALUE;
|
||||||
|
}
|
||||||
|
function h() {
|
||||||
|
return 42;
|
||||||
|
}
|
||||||
|
if (0)
|
||||||
|
console.debug("foo");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
expanded: {
|
||||||
|
options = {
|
||||||
|
global_defs: {
|
||||||
|
"CONFIG.DEBUG": [ 0 ],
|
||||||
|
"CONFIG.VALUE": 42,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
function f(CONFIG) {
|
||||||
|
// CONFIG not global - do not replace
|
||||||
|
return CONFIG.VALUE;
|
||||||
|
}
|
||||||
|
function g() {
|
||||||
|
var CONFIG = { VALUE: 1 };
|
||||||
|
// CONFIG not global - do not replace
|
||||||
|
return CONFIG.VALUE;
|
||||||
|
}
|
||||||
|
function h() {
|
||||||
|
return CONFIG.VALUE;
|
||||||
|
}
|
||||||
|
if (CONFIG.DEBUG[0])
|
||||||
|
console.debug("foo");
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
function f(CONFIG) {
|
||||||
|
return CONFIG.VALUE;
|
||||||
|
}
|
||||||
|
function g() {
|
||||||
|
var CONFIG = { VALUE: 1 };
|
||||||
|
return CONFIG.VALUE;
|
||||||
|
}
|
||||||
|
function h() {
|
||||||
|
return 42;
|
||||||
|
}
|
||||||
|
if ([0][0])
|
||||||
|
console.debug("foo");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
mixed: {
|
||||||
|
options = {
|
||||||
|
evaluate: true,
|
||||||
|
global_defs: {
|
||||||
|
"CONFIG.VALUE": 42,
|
||||||
|
"FOO.BAR": "moo",
|
||||||
|
},
|
||||||
|
properties: true,
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
const FOO = { BAR: 0 };
|
||||||
|
console.log(FOO.BAR);
|
||||||
|
console.log(++CONFIG.DEBUG);
|
||||||
|
console.log(++CONFIG.VALUE);
|
||||||
|
console.log(++CONFIG["VAL" + "UE"]);
|
||||||
|
console.log(++DEBUG[CONFIG.VALUE]);
|
||||||
|
CONFIG.VALUE.FOO = "bar";
|
||||||
|
console.log(CONFIG);
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
const FOO = { BAR: 0 };
|
||||||
|
console.log("moo");
|
||||||
|
console.log(++CONFIG.DEBUG);
|
||||||
|
console.log(++CONFIG.VALUE);
|
||||||
|
console.log(++CONFIG.VALUE);
|
||||||
|
console.log(++DEBUG[42]);
|
||||||
|
CONFIG.VALUE.FOO = "bar";
|
||||||
|
console.log(CONFIG);
|
||||||
|
}
|
||||||
|
expect_warnings: [
|
||||||
|
'WARN: global_defs CONFIG.VALUE redefined [test/compress/global_defs.js:126,22]',
|
||||||
|
'WARN: global_defs CONFIG.VALUE redefined [test/compress/global_defs.js:127,22]',
|
||||||
|
'WARN: global_defs CONFIG.VALUE redefined [test/compress/global_defs.js:129,8]',
|
||||||
|
]
|
||||||
|
}
|
||||||
90
test/compress/hoist_vars.js
Normal file
90
test/compress/hoist_vars.js
Normal file
@@ -0,0 +1,90 @@
|
|||||||
|
statements: {
|
||||||
|
options = {
|
||||||
|
hoist_funs: false,
|
||||||
|
hoist_vars: true,
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
function f() {
|
||||||
|
var a = 1;
|
||||||
|
var b = 2;
|
||||||
|
var c = 3;
|
||||||
|
function g() {}
|
||||||
|
return g(a, b, c);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
function f() {
|
||||||
|
var a = 1, b = 2, c = 3;
|
||||||
|
function g() {}
|
||||||
|
return g(a, b, c);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
statements_funs: {
|
||||||
|
options = {
|
||||||
|
hoist_funs: true,
|
||||||
|
hoist_vars: true,
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
function f() {
|
||||||
|
var a = 1;
|
||||||
|
var b = 2;
|
||||||
|
var c = 3;
|
||||||
|
function g() {}
|
||||||
|
return g(a, b, c);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
function f() {
|
||||||
|
function g() {}
|
||||||
|
var a = 1, b = 2, c = 3;
|
||||||
|
return g(a, b, c);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
sequences: {
|
||||||
|
options = {
|
||||||
|
hoist_funs: false,
|
||||||
|
hoist_vars: true,
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
function f() {
|
||||||
|
var a = 1, b = 2;
|
||||||
|
function g() {}
|
||||||
|
var c = 3;
|
||||||
|
return g(a, b, c);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
function f() {
|
||||||
|
var c, a = 1, b = 2;
|
||||||
|
function g() {}
|
||||||
|
c = 3;
|
||||||
|
return g(a, b, c);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
sequences_funs: {
|
||||||
|
options = {
|
||||||
|
hoist_funs: true,
|
||||||
|
hoist_vars: true,
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
function f() {
|
||||||
|
var a = 1, b = 2;
|
||||||
|
function g() {}
|
||||||
|
var c = 3;
|
||||||
|
return g(a, b, c);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
function f() {
|
||||||
|
function g() {}
|
||||||
|
var a = 1, b = 2, c = 3;
|
||||||
|
return g(a, b, c);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -170,8 +170,51 @@ if_return_7: {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
expect: {
|
expect: {
|
||||||
// suboptimal
|
function f(x){if(x)return!0;foo(),bar()}
|
||||||
function f(x){return!!x||(foo(),void bar())}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if_return_8: {
|
||||||
|
options = {
|
||||||
|
if_return: true,
|
||||||
|
sequences: true,
|
||||||
|
conditionals: true,
|
||||||
|
side_effects : true,
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
function f(e) {
|
||||||
|
if (2 == e) return foo();
|
||||||
|
if (3 == e) return bar();
|
||||||
|
if (4 == e) return baz();
|
||||||
|
fail(e);
|
||||||
|
}
|
||||||
|
|
||||||
|
function g(e) {
|
||||||
|
if (a(e)) return foo();
|
||||||
|
if (b(e)) return bar();
|
||||||
|
if (c(e)) return baz();
|
||||||
|
fail(e);
|
||||||
|
}
|
||||||
|
|
||||||
|
function h(e) {
|
||||||
|
if (a(e)) return foo();
|
||||||
|
else if (b(e)) return bar();
|
||||||
|
else if (c(e)) return baz();
|
||||||
|
else fail(e);
|
||||||
|
}
|
||||||
|
|
||||||
|
function i(e) {
|
||||||
|
if (a(e)) return foo();
|
||||||
|
else if (b(e)) return bar();
|
||||||
|
else if (c(e)) return baz();
|
||||||
|
fail(e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
function f(e){return 2==e?foo():3==e?bar():4==e?baz():void fail(e)}
|
||||||
|
function g(e){return a(e)?foo():b(e)?bar():c(e)?baz():void fail(e)}
|
||||||
|
function h(e){return a(e)?foo():b(e)?bar():c(e)?baz():void fail(e)}
|
||||||
|
function i(e){return a(e)?foo():b(e)?bar():c(e)?baz():void fail(e)}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -205,3 +248,57 @@ issue_1089: {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
issue_1437: {
|
||||||
|
options = {
|
||||||
|
if_return : true,
|
||||||
|
sequences : true,
|
||||||
|
conditionals : false
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
function x() {
|
||||||
|
if (a())
|
||||||
|
return b();
|
||||||
|
if (c())
|
||||||
|
return d();
|
||||||
|
else
|
||||||
|
e();
|
||||||
|
f();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
function x() {
|
||||||
|
if (a())
|
||||||
|
return b();
|
||||||
|
if (c())
|
||||||
|
return d();
|
||||||
|
else
|
||||||
|
e()
|
||||||
|
f();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
issue_1437_conditionals: {
|
||||||
|
options = {
|
||||||
|
conditionals : true,
|
||||||
|
if_return : true,
|
||||||
|
sequences : true
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
function x() {
|
||||||
|
if (a())
|
||||||
|
return b();
|
||||||
|
if (c())
|
||||||
|
return d();
|
||||||
|
else
|
||||||
|
e();
|
||||||
|
f();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
function x() {
|
||||||
|
return a() ? b() : c() ? d() : (e(), f(), void 0);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|||||||
@@ -13,7 +13,8 @@ const_declaration: {
|
|||||||
|
|
||||||
const_pragma: {
|
const_pragma: {
|
||||||
options = {
|
options = {
|
||||||
evaluate: true
|
evaluate: true,
|
||||||
|
reduce_vars: true,
|
||||||
};
|
};
|
||||||
|
|
||||||
input: {
|
input: {
|
||||||
@@ -27,7 +28,8 @@ const_pragma: {
|
|||||||
// for completeness' sake
|
// for completeness' sake
|
||||||
not_const: {
|
not_const: {
|
||||||
options = {
|
options = {
|
||||||
evaluate: true
|
evaluate: true,
|
||||||
|
reduce_vars: true,
|
||||||
};
|
};
|
||||||
|
|
||||||
input: {
|
input: {
|
||||||
|
|||||||
@@ -1,25 +0,0 @@
|
|||||||
typeof_eq_undefined: {
|
|
||||||
options = {
|
|
||||||
comparisons: true
|
|
||||||
};
|
|
||||||
input: { a = typeof b.c != "undefined" }
|
|
||||||
expect: { a = "undefined" != typeof b.c }
|
|
||||||
}
|
|
||||||
|
|
||||||
typeof_eq_undefined_unsafe: {
|
|
||||||
options = {
|
|
||||||
comparisons: true,
|
|
||||||
unsafe: true
|
|
||||||
};
|
|
||||||
input: { a = typeof b.c != "undefined" }
|
|
||||||
expect: { a = void 0 !== b.c }
|
|
||||||
}
|
|
||||||
|
|
||||||
typeof_eq_undefined_unsafe2: {
|
|
||||||
options = {
|
|
||||||
comparisons: true,
|
|
||||||
unsafe: true
|
|
||||||
};
|
|
||||||
input: { a = "undefined" != typeof b.c }
|
|
||||||
expect: { a = void 0 !== b.c }
|
|
||||||
}
|
|
||||||
52
test/compress/issue-1202.js
Normal file
52
test/compress/issue-1202.js
Normal file
@@ -0,0 +1,52 @@
|
|||||||
|
mangle_keep_fnames_false: {
|
||||||
|
options = {
|
||||||
|
keep_fnames : true,
|
||||||
|
keep_fargs : true,
|
||||||
|
}
|
||||||
|
mangle = {
|
||||||
|
keep_fnames : false,
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
"use strict";
|
||||||
|
function total() {
|
||||||
|
return function n(a, b, c) {
|
||||||
|
return a + b + c;
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
"use strict";
|
||||||
|
function total() {
|
||||||
|
return function t(n, r, u) {
|
||||||
|
return n + r + u;
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
mangle_keep_fnames_true: {
|
||||||
|
options = {
|
||||||
|
keep_fnames : true,
|
||||||
|
keep_fargs : true,
|
||||||
|
}
|
||||||
|
mangle = {
|
||||||
|
keep_fnames : true,
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
"use strict";
|
||||||
|
function total() {
|
||||||
|
return function n(a, b, c) {
|
||||||
|
return a + b + c;
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
"use strict";
|
||||||
|
function total() {
|
||||||
|
return function n(t, r, u) {
|
||||||
|
return t + r + u;
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
118
test/compress/issue-1261.js
Normal file
118
test/compress/issue-1261.js
Normal file
@@ -0,0 +1,118 @@
|
|||||||
|
pure_function_calls: {
|
||||||
|
options = {
|
||||||
|
evaluate : true,
|
||||||
|
conditionals : true,
|
||||||
|
comparisons : true,
|
||||||
|
side_effects : true,
|
||||||
|
booleans : true,
|
||||||
|
unused : true,
|
||||||
|
if_return : true,
|
||||||
|
join_vars : true,
|
||||||
|
cascade : true,
|
||||||
|
negate_iife : true,
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
// pure top-level IIFE will be dropped
|
||||||
|
// @__PURE__ - comment
|
||||||
|
(function() {
|
||||||
|
console.log("iife0");
|
||||||
|
})();
|
||||||
|
|
||||||
|
// pure top-level IIFE assigned to unreferenced var will not be dropped
|
||||||
|
var iife1 = /*@__PURE__*/(function() {
|
||||||
|
console.log("iife1");
|
||||||
|
function iife1() {}
|
||||||
|
return iife1;
|
||||||
|
})();
|
||||||
|
|
||||||
|
(function(){
|
||||||
|
// pure IIFE in function scope assigned to unreferenced var will be dropped
|
||||||
|
var iife2 = /*#__PURE__*/(function() {
|
||||||
|
console.log("iife2");
|
||||||
|
function iife2() {}
|
||||||
|
return iife2;
|
||||||
|
})();
|
||||||
|
})();
|
||||||
|
|
||||||
|
// comment #__PURE__ comment
|
||||||
|
bar(), baz(), quux();
|
||||||
|
a.b(), /* @__PURE__ */ c.d.e(), f.g();
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
var iife1 = function() {
|
||||||
|
console.log("iife1");
|
||||||
|
function iife1() {}
|
||||||
|
return iife1;
|
||||||
|
}();
|
||||||
|
|
||||||
|
baz(), quux();
|
||||||
|
a.b(), f.g();
|
||||||
|
}
|
||||||
|
expect_warnings: [
|
||||||
|
"WARN: Dropping __PURE__ call [test/compress/issue-1261.js:17,8]",
|
||||||
|
"WARN: Dropping side-effect-free statement [test/compress/issue-1261.js:17,8]",
|
||||||
|
"WARN: Dropping __PURE__ call [test/compress/issue-1261.js:30,37]",
|
||||||
|
"WARN: Dropping unused variable iife2 [test/compress/issue-1261.js:30,16]",
|
||||||
|
"WARN: Dropping side-effect-free statement [test/compress/issue-1261.js:28,8]",
|
||||||
|
"WARN: Dropping __PURE__ call [test/compress/issue-1261.js:38,8]",
|
||||||
|
"WARN: Dropping __PURE__ call [test/compress/issue-1261.js:39,31]",
|
||||||
|
]
|
||||||
|
}
|
||||||
|
|
||||||
|
pure_function_calls_toplevel: {
|
||||||
|
options = {
|
||||||
|
evaluate : true,
|
||||||
|
conditionals : true,
|
||||||
|
comparisons : true,
|
||||||
|
side_effects : true,
|
||||||
|
booleans : true,
|
||||||
|
unused : true,
|
||||||
|
if_return : true,
|
||||||
|
join_vars : true,
|
||||||
|
cascade : true,
|
||||||
|
negate_iife : true,
|
||||||
|
toplevel : true,
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
// pure top-level IIFE will be dropped
|
||||||
|
// @__PURE__ - comment
|
||||||
|
(function() {
|
||||||
|
console.log("iife0");
|
||||||
|
})();
|
||||||
|
|
||||||
|
// pure top-level IIFE assigned to unreferenced var will be dropped
|
||||||
|
var iife1 = /*@__PURE__*/(function() {
|
||||||
|
console.log("iife1");
|
||||||
|
function iife1() {}
|
||||||
|
return iife1;
|
||||||
|
})();
|
||||||
|
|
||||||
|
(function(){
|
||||||
|
// pure IIFE in function scope assigned to unreferenced var will be dropped
|
||||||
|
var iife2 = /*#__PURE__*/(function() {
|
||||||
|
console.log("iife2");
|
||||||
|
function iife2() {}
|
||||||
|
return iife2;
|
||||||
|
})();
|
||||||
|
})();
|
||||||
|
|
||||||
|
// comment #__PURE__ comment
|
||||||
|
bar(), baz(), quux();
|
||||||
|
a.b(), /* @__PURE__ */ c.d.e(), f.g();
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
baz(), quux();
|
||||||
|
a.b(), f.g();
|
||||||
|
}
|
||||||
|
expect_warnings: [
|
||||||
|
"WARN: Dropping __PURE__ call [test/compress/issue-1261.js:79,8]",
|
||||||
|
"WARN: Dropping side-effect-free statement [test/compress/issue-1261.js:79,8]",
|
||||||
|
"WARN: Dropping __PURE__ call [test/compress/issue-1261.js:92,37]",
|
||||||
|
"WARN: Dropping unused variable iife2 [test/compress/issue-1261.js:92,16]",
|
||||||
|
"WARN: Dropping side-effect-free statement [test/compress/issue-1261.js:90,8]",
|
||||||
|
"WARN: Dropping __PURE__ call [test/compress/issue-1261.js:100,8]",
|
||||||
|
"WARN: Dropping __PURE__ call [test/compress/issue-1261.js:101,31]",
|
||||||
|
"WARN: Dropping __PURE__ call [test/compress/issue-1261.js:84,33]",
|
||||||
|
"WARN: Dropping unused variable iife1 [test/compress/issue-1261.js:84,12]",
|
||||||
|
]
|
||||||
|
}
|
||||||
49
test/compress/issue-1275.js
Normal file
49
test/compress/issue-1275.js
Normal file
@@ -0,0 +1,49 @@
|
|||||||
|
string_plus_optimization: {
|
||||||
|
options = {
|
||||||
|
side_effects : true,
|
||||||
|
evaluate : true,
|
||||||
|
conditionals : true,
|
||||||
|
comparisons : true,
|
||||||
|
dead_code : true,
|
||||||
|
booleans : true,
|
||||||
|
unused : true,
|
||||||
|
if_return : true,
|
||||||
|
join_vars : true,
|
||||||
|
cascade : true,
|
||||||
|
hoist_funs : true,
|
||||||
|
};
|
||||||
|
input: {
|
||||||
|
function foo(anything) {
|
||||||
|
function throwing_function() {
|
||||||
|
throw "nope";
|
||||||
|
}
|
||||||
|
try {
|
||||||
|
console.log('0' + throwing_function() ? "yes" : "no");
|
||||||
|
} catch (ex) {
|
||||||
|
console.log(ex);
|
||||||
|
}
|
||||||
|
console.log('0' + anything ? "yes" : "no");
|
||||||
|
console.log(anything + '0' ? "Yes" : "No");
|
||||||
|
console.log('' + anything);
|
||||||
|
console.log(anything + '');
|
||||||
|
}
|
||||||
|
foo();
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
function foo(anything) {
|
||||||
|
function throwing_function() {
|
||||||
|
throw "nope";
|
||||||
|
}
|
||||||
|
try {
|
||||||
|
console.log('0' + throwing_function() ? "yes" : "no");
|
||||||
|
} catch (ex) {
|
||||||
|
console.log(ex);
|
||||||
|
}
|
||||||
|
console.log("yes");
|
||||||
|
console.log("Yes");
|
||||||
|
console.log('' + anything);
|
||||||
|
console.log(anything + '');
|
||||||
|
}
|
||||||
|
foo();
|
||||||
|
}
|
||||||
|
}
|
||||||
54
test/compress/issue-1321.js
Normal file
54
test/compress/issue-1321.js
Normal file
@@ -0,0 +1,54 @@
|
|||||||
|
issue_1321_no_debug: {
|
||||||
|
mangle_props = {
|
||||||
|
ignore_quoted: true
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
var x = {};
|
||||||
|
x.foo = 1;
|
||||||
|
x["a"] = 2 * x.foo;
|
||||||
|
console.log(x.foo, x["a"]);
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
var x = {};
|
||||||
|
x.b = 1;
|
||||||
|
x["a"] = 2 * x.b;
|
||||||
|
console.log(x.b, x["a"]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
issue_1321_debug: {
|
||||||
|
mangle_props = {
|
||||||
|
ignore_quoted: true,
|
||||||
|
debug: ""
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
var x = {};
|
||||||
|
x.foo = 1;
|
||||||
|
x["_$foo$_"] = 2 * x.foo;
|
||||||
|
console.log(x.foo, x["_$foo$_"]);
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
var x = {};
|
||||||
|
x.a = 1;
|
||||||
|
x["_$foo$_"] = 2 * x.a;
|
||||||
|
console.log(x.a, x["_$foo$_"]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
issue_1321_with_quoted: {
|
||||||
|
mangle_props = {
|
||||||
|
ignore_quoted: false
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
var x = {};
|
||||||
|
x.foo = 1;
|
||||||
|
x["a"] = 2 * x.foo;
|
||||||
|
console.log(x.foo, x["a"]);
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
var x = {};
|
||||||
|
x.a = 1;
|
||||||
|
x["b"] = 2 * x.a;
|
||||||
|
console.log(x.a, x["b"]);
|
||||||
|
}
|
||||||
|
}
|
||||||
151
test/compress/issue-1431.js
Normal file
151
test/compress/issue-1431.js
Normal file
@@ -0,0 +1,151 @@
|
|||||||
|
level_zero: {
|
||||||
|
options = {
|
||||||
|
keep_fnames: true
|
||||||
|
}
|
||||||
|
mangle = {
|
||||||
|
keep_fnames: true
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
function f(x) {
|
||||||
|
function n(a) {
|
||||||
|
return a * a;
|
||||||
|
}
|
||||||
|
return function() {
|
||||||
|
return x;
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
function f(r) {
|
||||||
|
function n(n) {
|
||||||
|
return n * n;
|
||||||
|
}
|
||||||
|
return function() {
|
||||||
|
return r;
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
level_one: {
|
||||||
|
options = {
|
||||||
|
keep_fnames: true
|
||||||
|
}
|
||||||
|
mangle = {
|
||||||
|
keep_fnames: true
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
function f(x) {
|
||||||
|
return function() {
|
||||||
|
function n(a) {
|
||||||
|
return a * a;
|
||||||
|
}
|
||||||
|
return x(n);
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
function f(r) {
|
||||||
|
return function() {
|
||||||
|
function n(n) {
|
||||||
|
return n * n;
|
||||||
|
}
|
||||||
|
return r(n);
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
level_two: {
|
||||||
|
options = {
|
||||||
|
keep_fnames: true
|
||||||
|
}
|
||||||
|
mangle = {
|
||||||
|
keep_fnames: true
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
function f(x) {
|
||||||
|
return function() {
|
||||||
|
function r(a) {
|
||||||
|
return a * a;
|
||||||
|
}
|
||||||
|
return function() {
|
||||||
|
function n(a) {
|
||||||
|
return a * a;
|
||||||
|
}
|
||||||
|
return x(n);
|
||||||
|
};
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
function f(t) {
|
||||||
|
return function() {
|
||||||
|
function r(n) {
|
||||||
|
return n * n;
|
||||||
|
}
|
||||||
|
return function() {
|
||||||
|
function n(n) {
|
||||||
|
return n * n;
|
||||||
|
}
|
||||||
|
return t(n);
|
||||||
|
};
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
level_three: {
|
||||||
|
options = {
|
||||||
|
keep_fnames: true
|
||||||
|
}
|
||||||
|
mangle = {
|
||||||
|
keep_fnames: true
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
function f(x) {
|
||||||
|
return function() {
|
||||||
|
function r(a) {
|
||||||
|
return a * a;
|
||||||
|
}
|
||||||
|
return [
|
||||||
|
function() {
|
||||||
|
function t(a) {
|
||||||
|
return a * a;
|
||||||
|
}
|
||||||
|
return t;
|
||||||
|
},
|
||||||
|
function() {
|
||||||
|
function n(a) {
|
||||||
|
return a * a;
|
||||||
|
}
|
||||||
|
return x(n);
|
||||||
|
}
|
||||||
|
];
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
function f(t) {
|
||||||
|
return function() {
|
||||||
|
function r(n) {
|
||||||
|
return n * n;
|
||||||
|
}
|
||||||
|
return [
|
||||||
|
function() {
|
||||||
|
function t(n) {
|
||||||
|
return n * n;
|
||||||
|
}
|
||||||
|
return t;
|
||||||
|
},
|
||||||
|
function() {
|
||||||
|
function n(n) {
|
||||||
|
return n * n;
|
||||||
|
}
|
||||||
|
return t(n);
|
||||||
|
}
|
||||||
|
];
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
69
test/compress/issue-1443.js
Normal file
69
test/compress/issue-1443.js
Normal file
@@ -0,0 +1,69 @@
|
|||||||
|
// tests assume that variable `undefined` not redefined and has `void 0` as value
|
||||||
|
|
||||||
|
unsafe_undefined: {
|
||||||
|
options = {
|
||||||
|
if_return: true,
|
||||||
|
unsafe: true
|
||||||
|
}
|
||||||
|
mangle = {}
|
||||||
|
input: {
|
||||||
|
function f(undefined) {
|
||||||
|
return function() {
|
||||||
|
if (a)
|
||||||
|
return b;
|
||||||
|
if (c)
|
||||||
|
return d;
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
function f(n) {
|
||||||
|
return function() {
|
||||||
|
if (a)
|
||||||
|
return b;
|
||||||
|
if (c)
|
||||||
|
return d;
|
||||||
|
else
|
||||||
|
return n;
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
keep_fnames: {
|
||||||
|
options = {
|
||||||
|
if_return: true,
|
||||||
|
unsafe: true
|
||||||
|
}
|
||||||
|
mangle = {
|
||||||
|
keep_fnames: true
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
function f(undefined) {
|
||||||
|
return function() {
|
||||||
|
function n(a) {
|
||||||
|
return a * a;
|
||||||
|
}
|
||||||
|
if (a)
|
||||||
|
return b;
|
||||||
|
if (c)
|
||||||
|
return d;
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
function f(r) {
|
||||||
|
return function() {
|
||||||
|
function n(n) {
|
||||||
|
return n * n;
|
||||||
|
}
|
||||||
|
if (a)
|
||||||
|
return b;
|
||||||
|
if (c)
|
||||||
|
return d;
|
||||||
|
else
|
||||||
|
return r;
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
71
test/compress/issue-1446.js
Normal file
71
test/compress/issue-1446.js
Normal file
@@ -0,0 +1,71 @@
|
|||||||
|
typeof_eq_undefined: {
|
||||||
|
options = {
|
||||||
|
comparisons: true
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
var a = typeof b != "undefined";
|
||||||
|
b = typeof a != "undefined";
|
||||||
|
var c = typeof d.e !== "undefined";
|
||||||
|
var f = "undefined" === typeof g;
|
||||||
|
g = "undefined" === typeof f;
|
||||||
|
var h = "undefined" == typeof i.j;
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
var a = "undefined" != typeof b;
|
||||||
|
b = void 0 !== a;
|
||||||
|
var c = void 0 !== d.e;
|
||||||
|
var f = "undefined" == typeof g;
|
||||||
|
g = void 0 === f;
|
||||||
|
var h = void 0 === i.j;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
typeof_eq_undefined_ie8: {
|
||||||
|
options = {
|
||||||
|
comparisons: true,
|
||||||
|
screw_ie8: false
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
var a = typeof b != "undefined";
|
||||||
|
b = typeof a != "undefined";
|
||||||
|
var c = typeof d.e !== "undefined";
|
||||||
|
var f = "undefined" === typeof g;
|
||||||
|
g = "undefined" === typeof f;
|
||||||
|
var h = "undefined" == typeof i.j;
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
var a = "undefined" != typeof b;
|
||||||
|
b = void 0 !== a;
|
||||||
|
var c = "undefined" != typeof d.e;
|
||||||
|
var f = "undefined" == typeof g;
|
||||||
|
g = void 0 === f;
|
||||||
|
var h = "undefined" == typeof i.j;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
undefined_redefined: {
|
||||||
|
options = {
|
||||||
|
comparisons: true
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
function f(undefined) {
|
||||||
|
var n = 1;
|
||||||
|
return typeof n == "undefined";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
expect_exact: "function f(undefined){var n=1;return void 0===n}"
|
||||||
|
}
|
||||||
|
|
||||||
|
undefined_redefined_mangle: {
|
||||||
|
options = {
|
||||||
|
comparisons: true
|
||||||
|
}
|
||||||
|
mangle = {}
|
||||||
|
input: {
|
||||||
|
function f(undefined) {
|
||||||
|
var n = 1;
|
||||||
|
return typeof n == "undefined";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
expect_exact: "function f(n){var r=1;return void 0===r}"
|
||||||
|
}
|
||||||
45
test/compress/issue-1447.js
Normal file
45
test/compress/issue-1447.js
Normal file
@@ -0,0 +1,45 @@
|
|||||||
|
else_with_empty_block: {
|
||||||
|
options = {}
|
||||||
|
input: {
|
||||||
|
if (x)
|
||||||
|
yes();
|
||||||
|
else {
|
||||||
|
}
|
||||||
|
}
|
||||||
|
expect_exact: "if(x)yes();"
|
||||||
|
}
|
||||||
|
|
||||||
|
else_with_empty_statement: {
|
||||||
|
options = {}
|
||||||
|
input: {
|
||||||
|
if (x)
|
||||||
|
yes();
|
||||||
|
else
|
||||||
|
;
|
||||||
|
}
|
||||||
|
expect_exact: "if(x)yes();"
|
||||||
|
}
|
||||||
|
|
||||||
|
conditional_false_stray_else_in_loop: {
|
||||||
|
options = {
|
||||||
|
evaluate : true,
|
||||||
|
comparisons : true,
|
||||||
|
booleans : true,
|
||||||
|
unused : true,
|
||||||
|
loops : true,
|
||||||
|
side_effects : true,
|
||||||
|
dead_code : true,
|
||||||
|
hoist_vars : true,
|
||||||
|
join_vars : true,
|
||||||
|
if_return : true,
|
||||||
|
cascade : true,
|
||||||
|
conditionals : false,
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
for (var i = 1; i <= 4; ++i) {
|
||||||
|
if (i <= 2) continue;
|
||||||
|
console.log(i);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
expect_exact: "for(var i=1;i<=4;++i)if(!(i<=2))console.log(i);"
|
||||||
|
}
|
||||||
@@ -1,11 +1,70 @@
|
|||||||
do_not_update_lhs: {
|
do_not_update_lhs: {
|
||||||
options = { global_defs: { DEBUG: false } };
|
options = {
|
||||||
input: { DEBUG = false; }
|
global_defs: { DEBUG: 0 }
|
||||||
expect: { DEBUG = false; }
|
}
|
||||||
|
input: {
|
||||||
|
DEBUG++;
|
||||||
|
DEBUG += 1;
|
||||||
|
DEBUG = 1;
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
DEBUG++;
|
||||||
|
DEBUG += 1;
|
||||||
|
DEBUG = 1;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
do_update_rhs: {
|
do_update_rhs: {
|
||||||
options = { global_defs: { DEBUG: false } };
|
options = {
|
||||||
input: { MY_DEBUG = DEBUG; }
|
global_defs: { DEBUG: 0 }
|
||||||
expect: { MY_DEBUG = false; }
|
}
|
||||||
|
input: {
|
||||||
|
MY_DEBUG = DEBUG;
|
||||||
|
MY_DEBUG += DEBUG;
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
MY_DEBUG = 0;
|
||||||
|
MY_DEBUG += 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
mixed: {
|
||||||
|
options = {
|
||||||
|
evaluate: true,
|
||||||
|
global_defs: {
|
||||||
|
DEBUG: 0,
|
||||||
|
ENV: 1,
|
||||||
|
FOO: 2,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
const ENV = 3;
|
||||||
|
var FOO = 4;
|
||||||
|
f(ENV * 10);
|
||||||
|
--FOO;
|
||||||
|
DEBUG = 1;
|
||||||
|
DEBUG++;
|
||||||
|
DEBUG += 1;
|
||||||
|
f(DEBUG);
|
||||||
|
x = DEBUG;
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
const ENV = 3;
|
||||||
|
var FOO = 4;
|
||||||
|
f(10);
|
||||||
|
--FOO;
|
||||||
|
DEBUG = 1;
|
||||||
|
DEBUG++;
|
||||||
|
DEBUG += 1;
|
||||||
|
f(0);
|
||||||
|
x = 0;
|
||||||
|
}
|
||||||
|
expect_warnings: [
|
||||||
|
'WARN: global_defs ENV redefined [test/compress/issue-208.js:41,14]',
|
||||||
|
'WARN: global_defs FOO redefined [test/compress/issue-208.js:42,12]',
|
||||||
|
'WARN: global_defs FOO redefined [test/compress/issue-208.js:44,10]',
|
||||||
|
'WARN: global_defs DEBUG redefined [test/compress/issue-208.js:45,8]',
|
||||||
|
'WARN: global_defs DEBUG redefined [test/compress/issue-208.js:46,8]',
|
||||||
|
'WARN: global_defs DEBUG redefined [test/compress/issue-208.js:47,8]',
|
||||||
|
]
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -82,7 +82,7 @@ issue979_test_negated_is_best: {
|
|||||||
1!=a||2!=b||foo();
|
1!=a||2!=b||foo();
|
||||||
}
|
}
|
||||||
function f7() {
|
function f7() {
|
||||||
return 1!=a&&2!=b?bar():void foo();
|
if(1!=a&&2!=b)return bar();foo()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -144,4 +144,75 @@ parse_do_while_without_semicolon: {
|
|||||||
expect: {
|
expect: {
|
||||||
do x(); while (false);y();
|
do x(); while (false);y();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
keep_collapse_const_in_own_block_scope: {
|
||||||
|
options = {
|
||||||
|
join_vars: true,
|
||||||
|
loops: true
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
var i=2;
|
||||||
|
const c=5;
|
||||||
|
while(i--)
|
||||||
|
console.log(i);
|
||||||
|
console.log(c);
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
var i=2;
|
||||||
|
const c=5;
|
||||||
|
for(;i--;)
|
||||||
|
console.log(i);
|
||||||
|
console.log(c);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
keep_collapse_const_in_own_block_scope_2: {
|
||||||
|
options = {
|
||||||
|
join_vars: true,
|
||||||
|
loops: true
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
const c=5;
|
||||||
|
var i=2; // Moves to loop, while it did not in previous test
|
||||||
|
while(i--)
|
||||||
|
console.log(i);
|
||||||
|
console.log(c);
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
const c=5;
|
||||||
|
for(var i=2;i--;)
|
||||||
|
console.log(i);
|
||||||
|
console.log(c);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
evaluate: {
|
||||||
|
options = {
|
||||||
|
loops: true,
|
||||||
|
dead_code: true,
|
||||||
|
evaluate: true,
|
||||||
|
};
|
||||||
|
input: {
|
||||||
|
while (true) {
|
||||||
|
a();
|
||||||
|
}
|
||||||
|
while (false) {
|
||||||
|
b();
|
||||||
|
}
|
||||||
|
do {
|
||||||
|
c();
|
||||||
|
} while (true);
|
||||||
|
do {
|
||||||
|
d();
|
||||||
|
} while (false);
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
for(;;)
|
||||||
|
a();
|
||||||
|
for(;;)
|
||||||
|
c();
|
||||||
|
d();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|||||||
28
test/compress/max_line_len.js
Normal file
28
test/compress/max_line_len.js
Normal file
@@ -0,0 +1,28 @@
|
|||||||
|
too_short: {
|
||||||
|
beautify = {
|
||||||
|
max_line_len: 10,
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
function f(a) {
|
||||||
|
return { c: 42, d: a(), e: "foo"};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
expect_exact: 'function f(a){\nreturn{\nc:42,\nd:a(),\ne:"foo"}}'
|
||||||
|
expect_warnings: [
|
||||||
|
"WARN: Output exceeds 10 characters"
|
||||||
|
]
|
||||||
|
}
|
||||||
|
|
||||||
|
just_enough: {
|
||||||
|
beautify = {
|
||||||
|
max_line_len: 14,
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
function f(a) {
|
||||||
|
return { c: 42, d: a(), e: "foo"};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
expect_exact: 'function f(a){\nreturn{c:42,\nd:a(),e:"foo"}\n}'
|
||||||
|
expect_warnings: [
|
||||||
|
]
|
||||||
|
}
|
||||||
@@ -10,6 +10,16 @@ negate_iife_1: {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
negate_iife_1_off: {
|
||||||
|
options = {
|
||||||
|
negate_iife: false,
|
||||||
|
};
|
||||||
|
input: {
|
||||||
|
(function(){ stuff() })();
|
||||||
|
}
|
||||||
|
expect_exact: '(function(){stuff()})();'
|
||||||
|
}
|
||||||
|
|
||||||
negate_iife_2: {
|
negate_iife_2: {
|
||||||
options = {
|
options = {
|
||||||
negate_iife: true
|
negate_iife: true
|
||||||
@@ -25,6 +35,7 @@ negate_iife_2: {
|
|||||||
negate_iife_3: {
|
negate_iife_3: {
|
||||||
options = {
|
options = {
|
||||||
negate_iife: true,
|
negate_iife: true,
|
||||||
|
conditionals: true
|
||||||
};
|
};
|
||||||
input: {
|
input: {
|
||||||
(function(){ return true })() ? console.log(true) : console.log(false);
|
(function(){ return true })() ? console.log(true) : console.log(false);
|
||||||
@@ -34,9 +45,23 @@ negate_iife_3: {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
negate_iife_3: {
|
negate_iife_3_off: {
|
||||||
|
options = {
|
||||||
|
negate_iife: false,
|
||||||
|
conditionals: true,
|
||||||
|
};
|
||||||
|
input: {
|
||||||
|
(function(){ return true })() ? console.log(true) : console.log(false);
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
!function(){ return true }() ? console.log(false) : console.log(true);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
negate_iife_4: {
|
||||||
options = {
|
options = {
|
||||||
negate_iife: true,
|
negate_iife: true,
|
||||||
|
conditionals: true,
|
||||||
sequences: true
|
sequences: true
|
||||||
};
|
};
|
||||||
input: {
|
input: {
|
||||||
@@ -52,7 +77,42 @@ negate_iife_3: {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
negate_iife_4: {
|
sequence_off: {
|
||||||
|
options = {
|
||||||
|
negate_iife: false,
|
||||||
|
conditionals: true,
|
||||||
|
sequences: true,
|
||||||
|
passes: 2,
|
||||||
|
};
|
||||||
|
input: {
|
||||||
|
function f() {
|
||||||
|
(function(){ return true })() ? console.log(true) : console.log(false);
|
||||||
|
(function(){
|
||||||
|
console.log("something");
|
||||||
|
})();
|
||||||
|
}
|
||||||
|
function g() {
|
||||||
|
(function(){
|
||||||
|
console.log("something");
|
||||||
|
})();
|
||||||
|
(function(){ return true })() ? console.log(true) : console.log(false);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
function f() {
|
||||||
|
!function(){ return true }() ? console.log(false) : console.log(true), function(){
|
||||||
|
console.log("something");
|
||||||
|
}();
|
||||||
|
}
|
||||||
|
function g() {
|
||||||
|
(function(){
|
||||||
|
console.log("something");
|
||||||
|
})(), function(){ return true }() ? console.log(true) : console.log(false);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
negate_iife_5: {
|
||||||
options = {
|
options = {
|
||||||
negate_iife: true,
|
negate_iife: true,
|
||||||
sequences: true,
|
sequences: true,
|
||||||
@@ -75,6 +135,29 @@ negate_iife_4: {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
negate_iife_5_off: {
|
||||||
|
options = {
|
||||||
|
negate_iife: false,
|
||||||
|
sequences: true,
|
||||||
|
conditionals: true,
|
||||||
|
};
|
||||||
|
input: {
|
||||||
|
if ((function(){ return true })()) {
|
||||||
|
foo(true);
|
||||||
|
} else {
|
||||||
|
bar(false);
|
||||||
|
}
|
||||||
|
(function(){
|
||||||
|
console.log("something");
|
||||||
|
})();
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
!function(){ return true }() ? bar(false) : foo(true), function(){
|
||||||
|
console.log("something");
|
||||||
|
}();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
negate_iife_nested: {
|
negate_iife_nested: {
|
||||||
options = {
|
options = {
|
||||||
negate_iife: true,
|
negate_iife: true,
|
||||||
@@ -107,6 +190,38 @@ negate_iife_nested: {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
negate_iife_nested_off: {
|
||||||
|
options = {
|
||||||
|
negate_iife: false,
|
||||||
|
sequences: true,
|
||||||
|
conditionals: true,
|
||||||
|
};
|
||||||
|
input: {
|
||||||
|
function Foo(f) {
|
||||||
|
this.f = f;
|
||||||
|
}
|
||||||
|
new Foo(function() {
|
||||||
|
(function(x) {
|
||||||
|
(function(y) {
|
||||||
|
console.log(y);
|
||||||
|
})(x);
|
||||||
|
})(7);
|
||||||
|
}).f();
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
function Foo(f) {
|
||||||
|
this.f = f;
|
||||||
|
}
|
||||||
|
new Foo(function() {
|
||||||
|
(function(x) {
|
||||||
|
(function(y) {
|
||||||
|
console.log(y);
|
||||||
|
})(x);
|
||||||
|
})(7);
|
||||||
|
}).f();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
negate_iife_issue_1073: {
|
negate_iife_issue_1073: {
|
||||||
options = {
|
options = {
|
||||||
negate_iife: true,
|
negate_iife: true,
|
||||||
@@ -130,3 +245,78 @@ negate_iife_issue_1073: {
|
|||||||
}(7))();
|
}(7))();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
issue_1254_negate_iife_false: {
|
||||||
|
options = {
|
||||||
|
negate_iife: false,
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
(function() {
|
||||||
|
return function() {
|
||||||
|
console.log('test')
|
||||||
|
};
|
||||||
|
})()();
|
||||||
|
}
|
||||||
|
expect_exact: '(function(){return function(){console.log("test")}})()();'
|
||||||
|
}
|
||||||
|
|
||||||
|
issue_1254_negate_iife_true: {
|
||||||
|
options = {
|
||||||
|
negate_iife: true,
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
(function() {
|
||||||
|
return function() {
|
||||||
|
console.log('test')
|
||||||
|
};
|
||||||
|
})()();
|
||||||
|
}
|
||||||
|
expect_exact: '!function(){return function(){console.log("test")}}()();'
|
||||||
|
}
|
||||||
|
|
||||||
|
issue_1254_negate_iife_nested: {
|
||||||
|
options = {
|
||||||
|
negate_iife: true,
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
(function() {
|
||||||
|
return function() {
|
||||||
|
console.log('test')
|
||||||
|
};
|
||||||
|
})()()()()();
|
||||||
|
}
|
||||||
|
expect_exact: '!function(){return function(){console.log("test")}}()()()()();'
|
||||||
|
}
|
||||||
|
|
||||||
|
issue_1288: {
|
||||||
|
options = {
|
||||||
|
negate_iife: true,
|
||||||
|
conditionals: true,
|
||||||
|
};
|
||||||
|
input: {
|
||||||
|
if (w) ;
|
||||||
|
else {
|
||||||
|
(function f() {})();
|
||||||
|
}
|
||||||
|
if (!x) {
|
||||||
|
(function() {
|
||||||
|
x = {};
|
||||||
|
})();
|
||||||
|
}
|
||||||
|
if (y)
|
||||||
|
(function() {})();
|
||||||
|
else
|
||||||
|
(function(z) {
|
||||||
|
return z;
|
||||||
|
})(0);
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
w || function f() {}();
|
||||||
|
x || function() {
|
||||||
|
x = {};
|
||||||
|
}();
|
||||||
|
y ? function() {}() : function(z) {
|
||||||
|
return z;
|
||||||
|
}(0);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|||||||
@@ -50,3 +50,35 @@ new_with_many_parameters: {
|
|||||||
}
|
}
|
||||||
expect_exact: 'new foo.bar("baz");new x(/123/,456);'
|
expect_exact: 'new foo.bar("baz");new x(/123/,456);'
|
||||||
}
|
}
|
||||||
|
|
||||||
|
new_constructor_with_unary_arguments: {
|
||||||
|
input: {
|
||||||
|
new x();
|
||||||
|
new x(-1);
|
||||||
|
new x(-1, -2);
|
||||||
|
new x(void 1, +2, -3, ~4, !5, --a, ++b, c--, d++, typeof e, delete f);
|
||||||
|
new (-1); // should parse despite being invalid at runtime.
|
||||||
|
new (-1)(); // should parse despite being invalid at runtime.
|
||||||
|
new (-1)(-2); // should parse despite being invalid at runtime.
|
||||||
|
}
|
||||||
|
expect_exact: "new x;new x(-1);new x(-1,-2);new x(void 1,+2,-3,~4,!5,--a,++b,c--,d++,typeof e,delete f);new(-1);new(-1);new(-1)(-2);"
|
||||||
|
}
|
||||||
|
|
||||||
|
call_with_unary_arguments: {
|
||||||
|
input: {
|
||||||
|
x();
|
||||||
|
x(-1);
|
||||||
|
x(-1, -2);
|
||||||
|
x(void 1, +2, -3, ~4, !5, --a, ++b, c--, d++, typeof e, delete f);
|
||||||
|
(-1)(); // should parse despite being invalid at runtime.
|
||||||
|
(-1)(-2); // should parse despite being invalid at runtime.
|
||||||
|
}
|
||||||
|
expect_exact: "x();x(-1);x(-1,-2);x(void 1,+2,-3,~4,!5,--a,++b,c--,d++,typeof e,delete f);(-1)();(-1)(-2);"
|
||||||
|
}
|
||||||
|
|
||||||
|
new_with_unary_prefix: {
|
||||||
|
input: {
|
||||||
|
var bar = (+new Date()).toString(32);
|
||||||
|
}
|
||||||
|
expect_exact: 'var bar=(+new Date).toString(32);';
|
||||||
|
}
|
||||||
|
|||||||
@@ -12,7 +12,8 @@ keep_properties: {
|
|||||||
|
|
||||||
dot_properties: {
|
dot_properties: {
|
||||||
options = {
|
options = {
|
||||||
properties: true
|
properties: true,
|
||||||
|
screw_ie8: false
|
||||||
};
|
};
|
||||||
input: {
|
input: {
|
||||||
a["foo"] = "bar";
|
a["foo"] = "bar";
|
||||||
@@ -53,7 +54,56 @@ dot_properties_es5: {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
evaluate_length: {
|
sub_properties: {
|
||||||
|
options = {
|
||||||
|
evaluate: true,
|
||||||
|
properties: true
|
||||||
|
};
|
||||||
|
input: {
|
||||||
|
a[0] = 0;
|
||||||
|
a["0"] = 1;
|
||||||
|
a[3.14] = 2;
|
||||||
|
a["3" + ".14"] = 3;
|
||||||
|
a["i" + "f"] = 4;
|
||||||
|
a["foo" + " bar"] = 5;
|
||||||
|
a[0 / 0] = 6;
|
||||||
|
a[null] = 7;
|
||||||
|
a[undefined] = 8;
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
a[0] = 0;
|
||||||
|
a[0] = 1;
|
||||||
|
a[3.14] = 2;
|
||||||
|
a[3.14] = 3;
|
||||||
|
a.if = 4;
|
||||||
|
a["foo bar"] = 5;
|
||||||
|
a[NaN] = 6;
|
||||||
|
a[null] = 7;
|
||||||
|
a[void 0] = 8;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
evaluate_array_length: {
|
||||||
|
options = {
|
||||||
|
properties: true,
|
||||||
|
unsafe: true,
|
||||||
|
evaluate: true
|
||||||
|
};
|
||||||
|
input: {
|
||||||
|
a = [1, 2, 3].length;
|
||||||
|
a = [1, 2, 3].join()["len" + "gth"];
|
||||||
|
a = [1, 2, b].length;
|
||||||
|
a = [1, 2, 3].join(b).length;
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
a = 3;
|
||||||
|
a = 5;
|
||||||
|
a = [1, 2, b].length;
|
||||||
|
a = [1, 2, 3].join(b).length;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
evaluate_string_length: {
|
||||||
options = {
|
options = {
|
||||||
properties: true,
|
properties: true,
|
||||||
unsafe: true,
|
unsafe: true,
|
||||||
@@ -81,15 +131,22 @@ mangle_properties: {
|
|||||||
a["foo"] = "bar";
|
a["foo"] = "bar";
|
||||||
a.color = "red";
|
a.color = "red";
|
||||||
x = {"bar": 10};
|
x = {"bar": 10};
|
||||||
|
a.run(x.bar, a.foo);
|
||||||
|
a['run']({color: "blue", foo: "baz"});
|
||||||
}
|
}
|
||||||
expect: {
|
expect: {
|
||||||
a["a"] = "bar";
|
a["a"] = "bar";
|
||||||
a.b = "red";
|
a.b = "red";
|
||||||
x = {c: 10};
|
x = {c: 10};
|
||||||
|
a.d(x.c, a.a);
|
||||||
|
a['d']({b: "blue", a: "baz"});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
mangle_unquoted_properties: {
|
mangle_unquoted_properties: {
|
||||||
|
options = {
|
||||||
|
properties: false
|
||||||
|
}
|
||||||
mangle_props = {
|
mangle_props = {
|
||||||
ignore_quoted: true
|
ignore_quoted: true
|
||||||
}
|
}
|
||||||
@@ -99,27 +156,402 @@ mangle_unquoted_properties: {
|
|||||||
keep_quoted_props: true,
|
keep_quoted_props: true,
|
||||||
}
|
}
|
||||||
input: {
|
input: {
|
||||||
|
a.top = 1;
|
||||||
function f1() {
|
function f1() {
|
||||||
a["foo"] = "bar";
|
a["foo"] = "bar";
|
||||||
a.color = "red";
|
a.color = "red";
|
||||||
x = {"bar": 10};
|
a.stuff = 2;
|
||||||
|
x = {"bar": 10, size: 7};
|
||||||
|
a.size = 9;
|
||||||
}
|
}
|
||||||
function f2() {
|
function f2() {
|
||||||
a.foo = "bar";
|
a.foo = "bar";
|
||||||
a['color'] = "red";
|
a['color'] = "red";
|
||||||
x = {bar: 10};
|
x = {bar: 10, size: 7};
|
||||||
|
a.size = 9;
|
||||||
|
a.stuff = 3;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
expect: {
|
expect: {
|
||||||
|
a.a = 1;
|
||||||
function f1() {
|
function f1() {
|
||||||
a["foo"] = "bar";
|
a["foo"] = "bar";
|
||||||
a.a = "red";
|
a.color = "red";
|
||||||
x = {"bar": 10};
|
a.b = 2;
|
||||||
|
x = {"bar": 10, c: 7};
|
||||||
|
a.c = 9;
|
||||||
}
|
}
|
||||||
function f2() {
|
function f2() {
|
||||||
a.b = "bar";
|
a.foo = "bar";
|
||||||
a['color'] = "red";
|
a['color'] = "red";
|
||||||
x = {c: 10};
|
x = {bar: 10, c: 7};
|
||||||
|
a.c = 9;
|
||||||
|
a.b = 3;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
mangle_debug: {
|
||||||
|
mangle_props = {
|
||||||
|
debug: ""
|
||||||
|
};
|
||||||
|
input: {
|
||||||
|
a.foo = "bar";
|
||||||
|
x = { baz: "ban" };
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
a._$foo$_ = "bar";
|
||||||
|
x = { _$baz$_: "ban" };
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
mangle_debug_true: {
|
||||||
|
mangle_props = {
|
||||||
|
debug: true
|
||||||
|
};
|
||||||
|
input: {
|
||||||
|
a.foo = "bar";
|
||||||
|
x = { baz: "ban" };
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
a._$foo$_ = "bar";
|
||||||
|
x = { _$baz$_: "ban" };
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
mangle_debug_suffix: {
|
||||||
|
mangle_props = {
|
||||||
|
debug: "XYZ"
|
||||||
|
};
|
||||||
|
input: {
|
||||||
|
a.foo = "bar";
|
||||||
|
x = { baz: "ban" };
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
a._$foo$XYZ_ = "bar";
|
||||||
|
x = { _$baz$XYZ_: "ban" };
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
mangle_debug_suffix_ignore_quoted: {
|
||||||
|
options = {
|
||||||
|
properties: false
|
||||||
|
}
|
||||||
|
mangle_props = {
|
||||||
|
ignore_quoted: true,
|
||||||
|
debug: "XYZ",
|
||||||
|
reserved: []
|
||||||
|
}
|
||||||
|
beautify = {
|
||||||
|
beautify: false,
|
||||||
|
quote_style: 3,
|
||||||
|
keep_quoted_props: true,
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
a.top = 1;
|
||||||
|
function f1() {
|
||||||
|
a["foo"] = "bar";
|
||||||
|
a.color = "red";
|
||||||
|
a.stuff = 2;
|
||||||
|
x = {"bar": 10, size: 7};
|
||||||
|
a.size = 9;
|
||||||
|
}
|
||||||
|
function f2() {
|
||||||
|
a.foo = "bar";
|
||||||
|
a['color'] = "red";
|
||||||
|
x = {bar: 10, size: 7};
|
||||||
|
a.size = 9;
|
||||||
|
a.stuff = 3;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
a._$top$XYZ_ = 1;
|
||||||
|
function f1() {
|
||||||
|
a["foo"] = "bar";
|
||||||
|
a.color = "red";
|
||||||
|
a._$stuff$XYZ_ = 2;
|
||||||
|
x = {"bar": 10, _$size$XYZ_: 7};
|
||||||
|
a._$size$XYZ_ = 9;
|
||||||
|
}
|
||||||
|
function f2() {
|
||||||
|
a.foo = "bar";
|
||||||
|
a['color'] = "red";
|
||||||
|
x = {bar: 10, _$size$XYZ_: 7};
|
||||||
|
a._$size$XYZ_ = 9;
|
||||||
|
a._$stuff$XYZ_ = 3;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
first_256_chars_as_properties: {
|
||||||
|
beautify = {
|
||||||
|
ascii_only: true,
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
// Note: some of these unicode character keys are not visible on github.com
|
||||||
|
var o = {
|
||||||
|
"\0":0,"":1,"":2,"":3,"":4,"":5,"":6,"":7,"\b":8,
|
||||||
|
"\t":9,"\n":10,"\v":11,"\f":12,"\r":13,"":14,"":15,"":16,"":17,
|
||||||
|
"":18,"":19,"":20,"":21,"":22,"":23,"":24,"":25,"":26,
|
||||||
|
"":27,"":28,"":29,"":30,"":31," ":32,"!":33,'"':34,"#":35,
|
||||||
|
$:36,"%":37,"&":38,"'":39,"(":40,")":41,"*":42,"+":43,",":44,
|
||||||
|
"-":45,".":46,"/":47,"0":48,"1":49,"2":50,"3":51,"4":52,"5":53,"6":54,"7":55,
|
||||||
|
"8":56,"9":57,":":58,";":59,"<":60,"=":61,">":62,"?":63,"@":64,A:65,
|
||||||
|
B:66,C:67,D:68,E:69,F:70,G:71,H:72,I:73,J:74,K:75,L:76,M:77,N:78,
|
||||||
|
O:79,P:80,Q:81,R:82,S:83,T:84,U:85,V:86,W:87,X:88,Y:89,Z:90,"[":91,
|
||||||
|
"\\":92,"]":93,"^":94,_:95,"`":96,a:97,b:98,c:99,d:100,e:101,
|
||||||
|
f:102,g:103,h:104,i:105,j:106,k:107,l:108,m:109,n:110,o:111,p:112,
|
||||||
|
q:113,r:114,s:115,t:116,u:117,v:118,w:119,x:120,y:121,z:122,"{":123,
|
||||||
|
"|":124,"}":125,"~":126,"":127,"":128,"":129,"":130,"":131,
|
||||||
|
"":132,"
":133,"":134,"":135,"":136,"":137,"":138,"":139,
|
||||||
|
"":140,"":141,"":142,"":143,"":144,"":145,"":146,"":147,
|
||||||
|
"":148,"":149,"":150,"":151,"":152,"":153,"":154,"":155,
|
||||||
|
"":156,"":157,"":158,"":159," ":160,"¡":161,"¢":162,"£":163,
|
||||||
|
"¤":164,"¥":165,"¦":166,"§":167,"¨":168,"©":169,"ª":170,"«":171,
|
||||||
|
"¬":172,"":173,"®":174,"¯":175,"°":176,"±":177,"²":178,"³":179,
|
||||||
|
"´":180,"µ":181,"¶":182,"·":183,"¸":184,"¹":185,"º":186,"»":187,
|
||||||
|
"¼":188,"½":189,"¾":190,"¿":191,"À":192,"Á":193,"Â":194,"Ã":195,
|
||||||
|
"Ä":196,"Å":197,"Æ":198,"Ç":199,"È":200,"É":201,"Ê":202,"Ë":203,
|
||||||
|
"Ì":204,"Í":205,"Î":206,"Ï":207,"Ð":208,"Ñ":209,"Ò":210,"Ó":211,
|
||||||
|
"Ô":212,"Õ":213,"Ö":214,"×":215,"Ø":216,"Ù":217,"Ú":218,"Û":219,
|
||||||
|
"Ü":220,"Ý":221,"Þ":222,"ß":223,"à":224,"á":225,"â":226,"ã":227,
|
||||||
|
"ä":228,"å":229,"æ":230,"ç":231,"è":232,"é":233,"ê":234,"ë":235,
|
||||||
|
"ì":236,"í":237,"î":238,"ï":239,"ð":240,"ñ":241,"ò":242,"ó":243,
|
||||||
|
"ô":244,"õ":245,"ö":246,"÷":247,"ø":248,"ù":249,"ú":250,"û":251,
|
||||||
|
"ü":252,"ý":253,"þ":254,"ÿ":255
|
||||||
|
};
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
var o = {
|
||||||
|
"\0":0,"\x01":1,"\x02":2,"\x03":3,"\x04":4,"\x05":5,"\x06":6,
|
||||||
|
"\x07":7,"\b":8,"\t":9,"\n":10,"\v":11,"\f":12,"\r":13,"\x0e":14,
|
||||||
|
"\x0f":15,"\x10":16,"\x11":17,"\x12":18,"\x13":19,"\x14":20,"\x15":21,
|
||||||
|
"\x16":22,"\x17":23,"\x18":24,"\x19":25,"\x1a":26,"\x1b":27,"\x1c":28,
|
||||||
|
"\x1d":29,"\x1e":30,"\x1f":31," ":32,"!":33,'"':34,"#":35,$:36,
|
||||||
|
"%":37,"&":38,"'":39,"(":40,")":41,"*":42,"+":43,",":44,"-":45,
|
||||||
|
".":46,"/":47,0:48,1:49,2:50,3:51,4:52,5:53,6:54,7:55,8:56,9:57,
|
||||||
|
":":58,";":59,"<":60,"=":61,">":62,"?":63,"@":64,A:65,B:66,C:67,
|
||||||
|
D:68,E:69,F:70,G:71,H:72,I:73,J:74,K:75,L:76,M:77,N:78,O:79,P:80,
|
||||||
|
Q:81,R:82,S:83,T:84,U:85,V:86,W:87,X:88,Y:89,Z:90,"[":91,"\\":92,
|
||||||
|
"]":93,"^":94,_:95,"`":96,a:97,b:98,c:99,d:100,e:101,f:102,g:103,
|
||||||
|
h:104,i:105,j:106,k:107,l:108,m:109,n:110,o:111,p:112,q:113,r:114,
|
||||||
|
s:115,t:116,u:117,v:118,w:119,x:120,y:121,z:122,"{":123,"|":124,
|
||||||
|
"}":125,"~":126,"\x7f":127,"\x80":128,"\x81":129,"\x82":130,"\x83":131,
|
||||||
|
"\x84":132,"\x85":133,"\x86":134,"\x87":135,"\x88":136,"\x89":137,
|
||||||
|
"\x8a":138,"\x8b":139,"\x8c":140,"\x8d":141,"\x8e":142,"\x8f":143,
|
||||||
|
"\x90":144,"\x91":145,"\x92":146,"\x93":147,"\x94":148,"\x95":149,
|
||||||
|
"\x96":150,"\x97":151,"\x98":152,"\x99":153,"\x9a":154,"\x9b":155,
|
||||||
|
"\x9c":156,"\x9d":157,"\x9e":158,"\x9f":159,"\xa0":160,"\xa1":161,
|
||||||
|
"\xa2":162,"\xa3":163,"\xa4":164,"\xa5":165,"\xa6":166,"\xa7":167,
|
||||||
|
"\xa8":168,"\xa9":169,"\xaa":170,"\xab":171,"\xac":172,"\xad":173,
|
||||||
|
"\xae":174,"\xaf":175,"\xb0":176,"\xb1":177,"\xb2":178,"\xb3":179,
|
||||||
|
"\xb4":180,"\xb5":181,"\xb6":182,"\xb7":183,"\xb8":184,"\xb9":185,
|
||||||
|
"\xba":186,"\xbb":187,"\xbc":188,"\xbd":189,"\xbe":190,"\xbf":191,
|
||||||
|
"\xc0":192,"\xc1":193,"\xc2":194,"\xc3":195,"\xc4":196,"\xc5":197,
|
||||||
|
"\xc6":198,"\xc7":199,"\xc8":200,"\xc9":201,"\xca":202,"\xcb":203,
|
||||||
|
"\xcc":204,"\xcd":205,"\xce":206,"\xcf":207,"\xd0":208,"\xd1":209,
|
||||||
|
"\xd2":210,"\xd3":211,"\xd4":212,"\xd5":213,"\xd6":214,"\xd7":215,
|
||||||
|
"\xd8":216,"\xd9":217,"\xda":218,"\xdb":219,"\xdc":220,"\xdd":221,
|
||||||
|
"\xde":222,"\xdf":223,"\xe0":224,"\xe1":225,"\xe2":226,"\xe3":227,
|
||||||
|
"\xe4":228,"\xe5":229,"\xe6":230,"\xe7":231,"\xe8":232,"\xe9":233,
|
||||||
|
"\xea":234,"\xeb":235,"\xec":236,"\xed":237,"\xee":238,"\xef":239,
|
||||||
|
"\xf0":240,"\xf1":241,"\xf2":242,"\xf3":243,"\xf4":244,"\xf5":245,
|
||||||
|
"\xf6":246,"\xf7":247,"\xf8":248,"\xf9":249,"\xfa":250,"\xfb":251,
|
||||||
|
"\xfc":252,"\xfd":253,"\xfe":254,"\xff":255
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
first_256_unicode_chars_as_properties: {
|
||||||
|
input: {
|
||||||
|
var o = {
|
||||||
|
"\u0000": 0, "\u0001": 1, "\u0002": 2, "\u0003": 3, "\u0004": 4, "\u0005": 5,
|
||||||
|
"\u0006": 6, "\u0007": 7, "\u0008": 8, "\u0009": 9, "\u000A": 10, "\u000B": 11,
|
||||||
|
"\u000C": 12, "\u000D": 13, "\u000E": 14, "\u000F": 15, "\u0010": 16, "\u0011": 17,
|
||||||
|
"\u0012": 18, "\u0013": 19, "\u0014": 20, "\u0015": 21, "\u0016": 22, "\u0017": 23,
|
||||||
|
"\u0018": 24, "\u0019": 25, "\u001A": 26, "\u001B": 27, "\u001C": 28, "\u001D": 29,
|
||||||
|
"\u001E": 30, "\u001F": 31, "\u0020": 32, "\u0021": 33, "\u0022": 34, "\u0023": 35,
|
||||||
|
"\u0024": 36, "\u0025": 37, "\u0026": 38, "\u0027": 39, "\u0028": 40, "\u0029": 41,
|
||||||
|
"\u002A": 42, "\u002B": 43, "\u002C": 44, "\u002D": 45, "\u002E": 46, "\u002F": 47,
|
||||||
|
"\u0030": 48, "\u0031": 49, "\u0032": 50, "\u0033": 51, "\u0034": 52, "\u0035": 53,
|
||||||
|
"\u0036": 54, "\u0037": 55, "\u0038": 56, "\u0039": 57, "\u003A": 58, "\u003B": 59,
|
||||||
|
"\u003C": 60, "\u003D": 61, "\u003E": 62, "\u003F": 63, "\u0040": 64, "\u0041": 65,
|
||||||
|
"\u0042": 66, "\u0043": 67, "\u0044": 68, "\u0045": 69, "\u0046": 70, "\u0047": 71,
|
||||||
|
"\u0048": 72, "\u0049": 73, "\u004A": 74, "\u004B": 75, "\u004C": 76, "\u004D": 77,
|
||||||
|
"\u004E": 78, "\u004F": 79, "\u0050": 80, "\u0051": 81, "\u0052": 82, "\u0053": 83,
|
||||||
|
"\u0054": 84, "\u0055": 85, "\u0056": 86, "\u0057": 87, "\u0058": 88, "\u0059": 89,
|
||||||
|
"\u005A": 90, "\u005B": 91, "\u005C": 92, "\u005D": 93, "\u005E": 94, "\u005F": 95,
|
||||||
|
"\u0060": 96, "\u0061": 97, "\u0062": 98, "\u0063": 99, "\u0064": 100, "\u0065": 101,
|
||||||
|
"\u0066": 102, "\u0067": 103, "\u0068": 104, "\u0069": 105, "\u006A": 106, "\u006B": 107,
|
||||||
|
"\u006C": 108, "\u006D": 109, "\u006E": 110, "\u006F": 111, "\u0070": 112, "\u0071": 113,
|
||||||
|
"\u0072": 114, "\u0073": 115, "\u0074": 116, "\u0075": 117, "\u0076": 118, "\u0077": 119,
|
||||||
|
"\u0078": 120, "\u0079": 121, "\u007A": 122, "\u007B": 123, "\u007C": 124, "\u007D": 125,
|
||||||
|
"\u007E": 126, "\u007F": 127, "\u0080": 128, "\u0081": 129, "\u0082": 130, "\u0083": 131,
|
||||||
|
"\u0084": 132, "\u0085": 133, "\u0086": 134, "\u0087": 135, "\u0088": 136, "\u0089": 137,
|
||||||
|
"\u008A": 138, "\u008B": 139, "\u008C": 140, "\u008D": 141, "\u008E": 142, "\u008F": 143,
|
||||||
|
"\u0090": 144, "\u0091": 145, "\u0092": 146, "\u0093": 147, "\u0094": 148, "\u0095": 149,
|
||||||
|
"\u0096": 150, "\u0097": 151, "\u0098": 152, "\u0099": 153, "\u009A": 154, "\u009B": 155,
|
||||||
|
"\u009C": 156, "\u009D": 157, "\u009E": 158, "\u009F": 159, "\u00A0": 160, "\u00A1": 161,
|
||||||
|
"\u00A2": 162, "\u00A3": 163, "\u00A4": 164, "\u00A5": 165, "\u00A6": 166, "\u00A7": 167,
|
||||||
|
"\u00A8": 168, "\u00A9": 169, "\u00AA": 170, "\u00AB": 171, "\u00AC": 172, "\u00AD": 173,
|
||||||
|
"\u00AE": 174, "\u00AF": 175, "\u00B0": 176, "\u00B1": 177, "\u00B2": 178, "\u00B3": 179,
|
||||||
|
"\u00B4": 180, "\u00B5": 181, "\u00B6": 182, "\u00B7": 183, "\u00B8": 184, "\u00B9": 185,
|
||||||
|
"\u00BA": 186, "\u00BB": 187, "\u00BC": 188, "\u00BD": 189, "\u00BE": 190, "\u00BF": 191,
|
||||||
|
"\u00C0": 192, "\u00C1": 193, "\u00C2": 194, "\u00C3": 195, "\u00C4": 196, "\u00C5": 197,
|
||||||
|
"\u00C6": 198, "\u00C7": 199, "\u00C8": 200, "\u00C9": 201, "\u00CA": 202, "\u00CB": 203,
|
||||||
|
"\u00CC": 204, "\u00CD": 205, "\u00CE": 206, "\u00CF": 207, "\u00D0": 208, "\u00D1": 209,
|
||||||
|
"\u00D2": 210, "\u00D3": 211, "\u00D4": 212, "\u00D5": 213, "\u00D6": 214, "\u00D7": 215,
|
||||||
|
"\u00D8": 216, "\u00D9": 217, "\u00DA": 218, "\u00DB": 219, "\u00DC": 220, "\u00DD": 221,
|
||||||
|
"\u00DE": 222, "\u00DF": 223, "\u00E0": 224, "\u00E1": 225, "\u00E2": 226, "\u00E3": 227,
|
||||||
|
"\u00E4": 228, "\u00E5": 229, "\u00E6": 230, "\u00E7": 231, "\u00E8": 232, "\u00E9": 233,
|
||||||
|
"\u00EA": 234, "\u00EB": 235, "\u00EC": 236, "\u00ED": 237, "\u00EE": 238, "\u00EF": 239,
|
||||||
|
"\u00F0": 240, "\u00F1": 241, "\u00F2": 242, "\u00F3": 243, "\u00F4": 244, "\u00F5": 245,
|
||||||
|
"\u00F6": 246, "\u00F7": 247, "\u00F8": 248, "\u00F9": 249, "\u00FA": 250, "\u00FB": 251,
|
||||||
|
"\u00FC": 252, "\u00FD": 253, "\u00FE": 254, "\u00FF": 255
|
||||||
|
};
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
var o = {
|
||||||
|
"\0":0,"\x01":1,"\x02":2,"\x03":3,"\x04":4,"\x05":5,"\x06":6,
|
||||||
|
"\x07":7,"\b":8,"\t":9,"\n":10,"\v":11,"\f":12,"\r":13,"\x0e":14,
|
||||||
|
"\x0f":15,"\x10":16,"\x11":17,"\x12":18,"\x13":19,"\x14":20,"\x15":21,
|
||||||
|
"\x16":22,"\x17":23,"\x18":24,"\x19":25,"\x1a":26,"\x1b":27,"\x1c":28,
|
||||||
|
"\x1d":29,"\x1e":30,"\x1f":31," ":32,"!":33,'"':34,"#":35,$:36,
|
||||||
|
"%":37,"&":38,"'":39,"(":40,")":41,"*":42,"+":43,",":44,"-":45,
|
||||||
|
".":46,"/":47,0:48,1:49,2:50,3:51,4:52,5:53,6:54,7:55,8:56,9:57,
|
||||||
|
":":58,";":59,"<":60,"=":61,">":62,"?":63,"@":64,A:65,B:66,C:67,
|
||||||
|
D:68,E:69,F:70,G:71,H:72,I:73,J:74,K:75,L:76,M:77,N:78,O:79,P:80,
|
||||||
|
Q:81,R:82,S:83,T:84,U:85,V:86,W:87,X:88,Y:89,Z:90,"[":91,"\\":92,
|
||||||
|
"]":93,"^":94,_:95,"`":96,a:97,b:98,c:99,d:100,e:101,f:102,g:103,
|
||||||
|
h:104,i:105,j:106,k:107,l:108,m:109,n:110,o:111,p:112,q:113,r:114,
|
||||||
|
s:115,t:116,u:117,v:118,w:119,x:120,y:121,z:122,"{":123,"|":124,
|
||||||
|
"}":125,"~":126,"\x7f":127,"\x80":128,"\x81":129,"\x82":130,"\x83":131,
|
||||||
|
"\x84":132,"\x85":133,"\x86":134,"\x87":135,"\x88":136,"\x89":137,
|
||||||
|
"\x8a":138,"\x8b":139,"\x8c":140,"\x8d":141,"\x8e":142,"\x8f":143,
|
||||||
|
"\x90":144,"\x91":145,"\x92":146,"\x93":147,"\x94":148,"\x95":149,
|
||||||
|
"\x96":150,"\x97":151,"\x98":152,"\x99":153,"\x9a":154,"\x9b":155,
|
||||||
|
"\x9c":156,"\x9d":157,"\x9e":158,"\x9f":159,"\xa0":160,"\xa1":161,
|
||||||
|
"\xa2":162,"\xa3":163,"\xa4":164,"\xa5":165,"\xa6":166,"\xa7":167,
|
||||||
|
"\xa8":168,"\xa9":169,"\xaa":170,"\xab":171,"\xac":172,"\xad":173,
|
||||||
|
"\xae":174,"\xaf":175,"\xb0":176,"\xb1":177,"\xb2":178,"\xb3":179,
|
||||||
|
"\xb4":180,"\xb5":181,"\xb6":182,"\xb7":183,"\xb8":184,"\xb9":185,
|
||||||
|
"\xba":186,"\xbb":187,"\xbc":188,"\xbd":189,"\xbe":190,"\xbf":191,
|
||||||
|
"\xc0":192,"\xc1":193,"\xc2":194,"\xc3":195,"\xc4":196,"\xc5":197,
|
||||||
|
"\xc6":198,"\xc7":199,"\xc8":200,"\xc9":201,"\xca":202,"\xcb":203,
|
||||||
|
"\xcc":204,"\xcd":205,"\xce":206,"\xcf":207,"\xd0":208,"\xd1":209,
|
||||||
|
"\xd2":210,"\xd3":211,"\xd4":212,"\xd5":213,"\xd6":214,"\xd7":215,
|
||||||
|
"\xd8":216,"\xd9":217,"\xda":218,"\xdb":219,"\xdc":220,"\xdd":221,
|
||||||
|
"\xde":222,"\xdf":223,"\xe0":224,"\xe1":225,"\xe2":226,"\xe3":227,
|
||||||
|
"\xe4":228,"\xe5":229,"\xe6":230,"\xe7":231,"\xe8":232,"\xe9":233,
|
||||||
|
"\xea":234,"\xeb":235,"\xec":236,"\xed":237,"\xee":238,"\xef":239,
|
||||||
|
"\xf0":240,"\xf1":241,"\xf2":242,"\xf3":243,"\xf4":244,"\xf5":245,
|
||||||
|
"\xf6":246,"\xf7":247,"\xf8":248,"\xf9":249,"\xfa":250,"\xfb":251,
|
||||||
|
"\xfc":252,"\xfd":253,"\xfe":254,"\xff":255
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
first_256_hex_chars_as_properties: {
|
||||||
|
input: {
|
||||||
|
var o = {
|
||||||
|
"\x00": 0, "\x01": 1, "\x02": 2, "\x03": 3, "\x04": 4, "\x05": 5,
|
||||||
|
"\x06": 6, "\x07": 7, "\x08": 8, "\x09": 9, "\x0A": 10, "\x0B": 11,
|
||||||
|
"\x0C": 12, "\x0D": 13, "\x0E": 14, "\x0F": 15, "\x10": 16, "\x11": 17,
|
||||||
|
"\x12": 18, "\x13": 19, "\x14": 20, "\x15": 21, "\x16": 22, "\x17": 23,
|
||||||
|
"\x18": 24, "\x19": 25, "\x1A": 26, "\x1B": 27, "\x1C": 28, "\x1D": 29,
|
||||||
|
"\x1E": 30, "\x1F": 31, "\x20": 32, "\x21": 33, "\x22": 34, "\x23": 35,
|
||||||
|
"\x24": 36, "\x25": 37, "\x26": 38, "\x27": 39, "\x28": 40, "\x29": 41,
|
||||||
|
"\x2A": 42, "\x2B": 43, "\x2C": 44, "\x2D": 45, "\x2E": 46, "\x2F": 47,
|
||||||
|
"\x30": 48, "\x31": 49, "\x32": 50, "\x33": 51, "\x34": 52, "\x35": 53,
|
||||||
|
"\x36": 54, "\x37": 55, "\x38": 56, "\x39": 57, "\x3A": 58, "\x3B": 59,
|
||||||
|
"\x3C": 60, "\x3D": 61, "\x3E": 62, "\x3F": 63, "\x40": 64, "\x41": 65,
|
||||||
|
"\x42": 66, "\x43": 67, "\x44": 68, "\x45": 69, "\x46": 70, "\x47": 71,
|
||||||
|
"\x48": 72, "\x49": 73, "\x4A": 74, "\x4B": 75, "\x4C": 76, "\x4D": 77,
|
||||||
|
"\x4E": 78, "\x4F": 79, "\x50": 80, "\x51": 81, "\x52": 82, "\x53": 83,
|
||||||
|
"\x54": 84, "\x55": 85, "\x56": 86, "\x57": 87, "\x58": 88, "\x59": 89,
|
||||||
|
"\x5A": 90, "\x5B": 91, "\x5C": 92, "\x5D": 93, "\x5E": 94, "\x5F": 95,
|
||||||
|
"\x60": 96, "\x61": 97, "\x62": 98, "\x63": 99, "\x64": 100, "\x65": 101,
|
||||||
|
"\x66": 102, "\x67": 103, "\x68": 104, "\x69": 105, "\x6A": 106, "\x6B": 107,
|
||||||
|
"\x6C": 108, "\x6D": 109, "\x6E": 110, "\x6F": 111, "\x70": 112, "\x71": 113,
|
||||||
|
"\x72": 114, "\x73": 115, "\x74": 116, "\x75": 117, "\x76": 118, "\x77": 119,
|
||||||
|
"\x78": 120, "\x79": 121, "\x7A": 122, "\x7B": 123, "\x7C": 124, "\x7D": 125,
|
||||||
|
"\x7E": 126, "\x7F": 127, "\x80": 128, "\x81": 129, "\x82": 130, "\x83": 131,
|
||||||
|
"\x84": 132, "\x85": 133, "\x86": 134, "\x87": 135, "\x88": 136, "\x89": 137,
|
||||||
|
"\x8A": 138, "\x8B": 139, "\x8C": 140, "\x8D": 141, "\x8E": 142, "\x8F": 143,
|
||||||
|
"\x90": 144, "\x91": 145, "\x92": 146, "\x93": 147, "\x94": 148, "\x95": 149,
|
||||||
|
"\x96": 150, "\x97": 151, "\x98": 152, "\x99": 153, "\x9A": 154, "\x9B": 155,
|
||||||
|
"\x9C": 156, "\x9D": 157, "\x9E": 158, "\x9F": 159, "\xA0": 160, "\xA1": 161,
|
||||||
|
"\xA2": 162, "\xA3": 163, "\xA4": 164, "\xA5": 165, "\xA6": 166, "\xA7": 167,
|
||||||
|
"\xA8": 168, "\xA9": 169, "\xAA": 170, "\xAB": 171, "\xAC": 172, "\xAD": 173,
|
||||||
|
"\xAE": 174, "\xAF": 175, "\xB0": 176, "\xB1": 177, "\xB2": 178, "\xB3": 179,
|
||||||
|
"\xB4": 180, "\xB5": 181, "\xB6": 182, "\xB7": 183, "\xB8": 184, "\xB9": 185,
|
||||||
|
"\xBA": 186, "\xBB": 187, "\xBC": 188, "\xBD": 189, "\xBE": 190, "\xBF": 191,
|
||||||
|
"\xC0": 192, "\xC1": 193, "\xC2": 194, "\xC3": 195, "\xC4": 196, "\xC5": 197,
|
||||||
|
"\xC6": 198, "\xC7": 199, "\xC8": 200, "\xC9": 201, "\xCA": 202, "\xCB": 203,
|
||||||
|
"\xCC": 204, "\xCD": 205, "\xCE": 206, "\xCF": 207, "\xD0": 208, "\xD1": 209,
|
||||||
|
"\xD2": 210, "\xD3": 211, "\xD4": 212, "\xD5": 213, "\xD6": 214, "\xD7": 215,
|
||||||
|
"\xD8": 216, "\xD9": 217, "\xDA": 218, "\xDB": 219, "\xDC": 220, "\xDD": 221,
|
||||||
|
"\xDE": 222, "\xDF": 223, "\xE0": 224, "\xE1": 225, "\xE2": 226, "\xE3": 227,
|
||||||
|
"\xE4": 228, "\xE5": 229, "\xE6": 230, "\xE7": 231, "\xE8": 232, "\xE9": 233,
|
||||||
|
"\xEA": 234, "\xEB": 235, "\xEC": 236, "\xED": 237, "\xEE": 238, "\xEF": 239,
|
||||||
|
"\xF0": 240, "\xF1": 241, "\xF2": 242, "\xF3": 243, "\xF4": 244, "\xF5": 245,
|
||||||
|
"\xF6": 246, "\xF7": 247, "\xF8": 248, "\xF9": 249, "\xFA": 250, "\xFB": 251,
|
||||||
|
"\xFC": 252, "\xFD": 253, "\xFE": 254, "\xFF": 255
|
||||||
|
};
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
var o = {
|
||||||
|
"\0":0,"\x01":1,"\x02":2,"\x03":3,"\x04":4,"\x05":5,"\x06":6,
|
||||||
|
"\x07":7,"\b":8,"\t":9,"\n":10,"\v":11,"\f":12,"\r":13,"\x0e":14,
|
||||||
|
"\x0f":15,"\x10":16,"\x11":17,"\x12":18,"\x13":19,"\x14":20,"\x15":21,
|
||||||
|
"\x16":22,"\x17":23,"\x18":24,"\x19":25,"\x1a":26,"\x1b":27,"\x1c":28,
|
||||||
|
"\x1d":29,"\x1e":30,"\x1f":31," ":32,"!":33,'"':34,"#":35,$:36,
|
||||||
|
"%":37,"&":38,"'":39,"(":40,")":41,"*":42,"+":43,",":44,"-":45,
|
||||||
|
".":46,"/":47,0:48,1:49,2:50,3:51,4:52,5:53,6:54,7:55,8:56,9:57,
|
||||||
|
":":58,";":59,"<":60,"=":61,">":62,"?":63,"@":64,A:65,B:66,C:67,
|
||||||
|
D:68,E:69,F:70,G:71,H:72,I:73,J:74,K:75,L:76,M:77,N:78,O:79,P:80,
|
||||||
|
Q:81,R:82,S:83,T:84,U:85,V:86,W:87,X:88,Y:89,Z:90,"[":91,"\\":92,
|
||||||
|
"]":93,"^":94,_:95,"`":96,a:97,b:98,c:99,d:100,e:101,f:102,g:103,
|
||||||
|
h:104,i:105,j:106,k:107,l:108,m:109,n:110,o:111,p:112,q:113,r:114,
|
||||||
|
s:115,t:116,u:117,v:118,w:119,x:120,y:121,z:122,"{":123,"|":124,
|
||||||
|
"}":125,"~":126,"\x7f":127,"\x80":128,"\x81":129,"\x82":130,"\x83":131,
|
||||||
|
"\x84":132,"\x85":133,"\x86":134,"\x87":135,"\x88":136,"\x89":137,
|
||||||
|
"\x8a":138,"\x8b":139,"\x8c":140,"\x8d":141,"\x8e":142,"\x8f":143,
|
||||||
|
"\x90":144,"\x91":145,"\x92":146,"\x93":147,"\x94":148,"\x95":149,
|
||||||
|
"\x96":150,"\x97":151,"\x98":152,"\x99":153,"\x9a":154,"\x9b":155,
|
||||||
|
"\x9c":156,"\x9d":157,"\x9e":158,"\x9f":159,"\xa0":160,"\xa1":161,
|
||||||
|
"\xa2":162,"\xa3":163,"\xa4":164,"\xa5":165,"\xa6":166,"\xa7":167,
|
||||||
|
"\xa8":168,"\xa9":169,"\xaa":170,"\xab":171,"\xac":172,"\xad":173,
|
||||||
|
"\xae":174,"\xaf":175,"\xb0":176,"\xb1":177,"\xb2":178,"\xb3":179,
|
||||||
|
"\xb4":180,"\xb5":181,"\xb6":182,"\xb7":183,"\xb8":184,"\xb9":185,
|
||||||
|
"\xba":186,"\xbb":187,"\xbc":188,"\xbd":189,"\xbe":190,"\xbf":191,
|
||||||
|
"\xc0":192,"\xc1":193,"\xc2":194,"\xc3":195,"\xc4":196,"\xc5":197,
|
||||||
|
"\xc6":198,"\xc7":199,"\xc8":200,"\xc9":201,"\xca":202,"\xcb":203,
|
||||||
|
"\xcc":204,"\xcd":205,"\xce":206,"\xcf":207,"\xd0":208,"\xd1":209,
|
||||||
|
"\xd2":210,"\xd3":211,"\xd4":212,"\xd5":213,"\xd6":214,"\xd7":215,
|
||||||
|
"\xd8":216,"\xd9":217,"\xda":218,"\xdb":219,"\xdc":220,"\xdd":221,
|
||||||
|
"\xde":222,"\xdf":223,"\xe0":224,"\xe1":225,"\xe2":226,"\xe3":227,
|
||||||
|
"\xe4":228,"\xe5":229,"\xe6":230,"\xe7":231,"\xe8":232,"\xe9":233,
|
||||||
|
"\xea":234,"\xeb":235,"\xec":236,"\xed":237,"\xee":238,"\xef":239,
|
||||||
|
"\xf0":240,"\xf1":241,"\xf2":242,"\xf3":243,"\xf4":244,"\xf5":245,
|
||||||
|
"\xf6":246,"\xf7":247,"\xf8":248,"\xf9":249,"\xfa":250,"\xfb":251,
|
||||||
|
"\xfc":252,"\xfd":253,"\xfe":254,"\xff":255
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
native_prototype: {
|
||||||
|
options = {
|
||||||
|
unsafe_proto: true,
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
Array.prototype.splice.apply(a, [1, 2, b, c]);
|
||||||
|
Object.prototype.hasOwnProperty.call(d, "foo");
|
||||||
|
String.prototype.indexOf.call(e, "bar");
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
[].splice.apply(a, [1, 2, b, c]);
|
||||||
|
({}).hasOwnProperty.call(d, "foo");
|
||||||
|
"".indexOf.call(e, "bar");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|||||||
295
test/compress/pure_funcs.js
Normal file
295
test/compress/pure_funcs.js
Normal file
@@ -0,0 +1,295 @@
|
|||||||
|
array: {
|
||||||
|
options = {
|
||||||
|
pure_funcs: [ "Math.floor" ],
|
||||||
|
side_effects: true,
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
var a;
|
||||||
|
function f(b) {
|
||||||
|
Math.floor(a / b);
|
||||||
|
Math.floor(c / b);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
var a;
|
||||||
|
function f(b) {
|
||||||
|
c;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func: {
|
||||||
|
options = {
|
||||||
|
pure_funcs: function(node) {
|
||||||
|
return !~node.args[0].print_to_string().indexOf("a");
|
||||||
|
},
|
||||||
|
side_effects: true,
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
function f(a, b) {
|
||||||
|
Math.floor(a / b);
|
||||||
|
Math.floor(c / b);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
function f(a, b) {
|
||||||
|
Math.floor(c / b);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
side_effects: {
|
||||||
|
options = {
|
||||||
|
pure_funcs: [ "console.log" ],
|
||||||
|
side_effects: true,
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
function f(a, b) {
|
||||||
|
console.log(a());
|
||||||
|
console.log(b);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
function f(a, b) {
|
||||||
|
a();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
unused: {
|
||||||
|
options = {
|
||||||
|
pure_funcs: [ "pure" ],
|
||||||
|
side_effects: true,
|
||||||
|
unused: true,
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
function foo() {
|
||||||
|
var u = pure(1);
|
||||||
|
var x = pure(2);
|
||||||
|
var y = pure(x);
|
||||||
|
var z = pure(pure(side_effects()));
|
||||||
|
return pure(3);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
function foo() {
|
||||||
|
side_effects();
|
||||||
|
return pure(3);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
babel: {
|
||||||
|
options = {
|
||||||
|
pure_funcs: [ "_classCallCheck" ],
|
||||||
|
side_effects: true,
|
||||||
|
unused: true,
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
function _classCallCheck(instance, Constructor) {
|
||||||
|
if (!(instance instanceof Constructor))
|
||||||
|
throw new TypeError("Cannot call a class as a function");
|
||||||
|
}
|
||||||
|
var Foo = function Foo() {
|
||||||
|
_classCallCheck(this, Foo);
|
||||||
|
};
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
function _classCallCheck(instance, Constructor) {
|
||||||
|
if (!(instance instanceof Constructor))
|
||||||
|
throw new TypeError("Cannot call a class as a function");
|
||||||
|
}
|
||||||
|
var Foo = function() {
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
conditional: {
|
||||||
|
options = {
|
||||||
|
pure_funcs: [ "pure" ],
|
||||||
|
side_effects: true,
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
pure(1 | a() ? 2 & b() : 7 ^ c());
|
||||||
|
pure(1 | a() ? 2 & b() : 5);
|
||||||
|
pure(1 | a() ? 4 : 7 ^ c());
|
||||||
|
pure(1 | a() ? 4 : 5);
|
||||||
|
pure(3 ? 2 & b() : 7 ^ c());
|
||||||
|
pure(3 ? 2 & b() : 5);
|
||||||
|
pure(3 ? 4 : 7 ^ c());
|
||||||
|
pure(3 ? 4 : 5);
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
1 | a() ? b() : c();
|
||||||
|
1 | a() && b();
|
||||||
|
1 | a() || c();
|
||||||
|
a();
|
||||||
|
3 ? b() : c();
|
||||||
|
3 && b();
|
||||||
|
3 || c();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
relational: {
|
||||||
|
options = {
|
||||||
|
pure_funcs: [ "foo" ],
|
||||||
|
side_effects :true,
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
foo() in foo();
|
||||||
|
foo() instanceof bar();
|
||||||
|
foo() < "bar";
|
||||||
|
bar() > foo();
|
||||||
|
bar() != bar();
|
||||||
|
bar() !== "bar";
|
||||||
|
"bar" == foo();
|
||||||
|
"bar" === bar();
|
||||||
|
"bar" >= "bar";
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
bar();
|
||||||
|
bar();
|
||||||
|
bar(), bar();
|
||||||
|
bar();
|
||||||
|
bar();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
arithmetic: {
|
||||||
|
options = {
|
||||||
|
pure_funcs: [ "foo" ],
|
||||||
|
side_effects :true,
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
foo() + foo();
|
||||||
|
foo() - bar();
|
||||||
|
foo() * "bar";
|
||||||
|
bar() / foo();
|
||||||
|
bar() & bar();
|
||||||
|
bar() | "bar";
|
||||||
|
"bar" >> foo();
|
||||||
|
"bar" << bar();
|
||||||
|
"bar" >>> "bar";
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
bar();
|
||||||
|
bar();
|
||||||
|
bar(), bar();
|
||||||
|
bar();
|
||||||
|
bar();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
boolean_and: {
|
||||||
|
options = {
|
||||||
|
pure_funcs: [ "foo" ],
|
||||||
|
side_effects :true,
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
foo() && foo();
|
||||||
|
foo() && bar();
|
||||||
|
foo() && "bar";
|
||||||
|
bar() && foo();
|
||||||
|
bar() && bar();
|
||||||
|
bar() && "bar";
|
||||||
|
"bar" && foo();
|
||||||
|
"bar" && bar();
|
||||||
|
"bar" && "bar";
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
foo() && bar();
|
||||||
|
bar();
|
||||||
|
bar() && bar();
|
||||||
|
bar();
|
||||||
|
"bar" && bar();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
boolean_or: {
|
||||||
|
options = {
|
||||||
|
pure_funcs: [ "foo" ],
|
||||||
|
side_effects :true,
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
foo() || foo();
|
||||||
|
foo() || bar();
|
||||||
|
foo() || "bar";
|
||||||
|
bar() || foo();
|
||||||
|
bar() || bar();
|
||||||
|
bar() || "bar";
|
||||||
|
"bar" || foo();
|
||||||
|
"bar" || bar();
|
||||||
|
"bar" || "bar";
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
foo() || bar();
|
||||||
|
bar();
|
||||||
|
bar() || bar();
|
||||||
|
bar();
|
||||||
|
"bar" || bar();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
assign: {
|
||||||
|
options = {
|
||||||
|
pure_funcs: [ "foo" ],
|
||||||
|
side_effects :true,
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
var a;
|
||||||
|
function f(b) {
|
||||||
|
a = foo();
|
||||||
|
b *= 4 + foo();
|
||||||
|
c >>= 0 | foo();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
var a;
|
||||||
|
function f(b) {
|
||||||
|
a = foo();
|
||||||
|
b *= 4 + foo();
|
||||||
|
c >>= 0 | foo();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
unary: {
|
||||||
|
options = {
|
||||||
|
pure_funcs: [ "foo" ],
|
||||||
|
side_effects :true,
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
typeof foo();
|
||||||
|
typeof bar();
|
||||||
|
typeof "bar";
|
||||||
|
void foo();
|
||||||
|
void bar();
|
||||||
|
void "bar";
|
||||||
|
delete a[foo()];
|
||||||
|
delete a[bar()];
|
||||||
|
delete a["bar"];
|
||||||
|
a[foo()]++;
|
||||||
|
a[bar()]++;
|
||||||
|
a["bar"]++;
|
||||||
|
--a[foo()];
|
||||||
|
--a[bar()];
|
||||||
|
--a["bar"];
|
||||||
|
~foo();
|
||||||
|
~bar();
|
||||||
|
~"bar";
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
bar();
|
||||||
|
bar();
|
||||||
|
delete a[foo()];
|
||||||
|
delete a[bar()];
|
||||||
|
delete a["bar"];
|
||||||
|
a[foo()]++;
|
||||||
|
a[bar()]++;
|
||||||
|
a["bar"]++;
|
||||||
|
--a[foo()];
|
||||||
|
--a[bar()];
|
||||||
|
--a["bar"];
|
||||||
|
bar();
|
||||||
|
}
|
||||||
|
}
|
||||||
472
test/compress/reduce_vars.js
Normal file
472
test/compress/reduce_vars.js
Normal file
@@ -0,0 +1,472 @@
|
|||||||
|
reduce_vars: {
|
||||||
|
options = {
|
||||||
|
conditionals : true,
|
||||||
|
evaluate : true,
|
||||||
|
global_defs : {
|
||||||
|
C : 0
|
||||||
|
},
|
||||||
|
reduce_vars : true,
|
||||||
|
unused : true
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
var A = 1;
|
||||||
|
(function f0() {
|
||||||
|
var a = 2;
|
||||||
|
console.log(a - 5);
|
||||||
|
console.log(A - 5);
|
||||||
|
})();
|
||||||
|
(function f1() {
|
||||||
|
var a = 2;
|
||||||
|
console.log(a - 5);
|
||||||
|
eval("console.log(a);");
|
||||||
|
})();
|
||||||
|
(function f2(eval) {
|
||||||
|
var a = 2;
|
||||||
|
console.log(a - 5);
|
||||||
|
eval("console.log(a);");
|
||||||
|
})(eval);
|
||||||
|
(function f3() {
|
||||||
|
var b = typeof C !== "undefined";
|
||||||
|
var c = 4;
|
||||||
|
if (b) {
|
||||||
|
return 'yes';
|
||||||
|
} else {
|
||||||
|
return 'no';
|
||||||
|
}
|
||||||
|
})();
|
||||||
|
console.log(A + 1);
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
var A = 1;
|
||||||
|
(function() {
|
||||||
|
console.log(-3);
|
||||||
|
console.log(-4);
|
||||||
|
})();
|
||||||
|
(function f1() {
|
||||||
|
var a = 2;
|
||||||
|
console.log(-3);
|
||||||
|
eval("console.log(a);");
|
||||||
|
})();
|
||||||
|
(function f2(eval) {
|
||||||
|
var a = 2;
|
||||||
|
console.log(-3);
|
||||||
|
eval("console.log(a);");
|
||||||
|
})(eval);
|
||||||
|
(function() {
|
||||||
|
return "yes";
|
||||||
|
})();
|
||||||
|
console.log(2);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
modified: {
|
||||||
|
options = {
|
||||||
|
conditionals : true,
|
||||||
|
evaluate : true,
|
||||||
|
reduce_vars : true,
|
||||||
|
unused : true
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
function f0() {
|
||||||
|
var a = 1, b = 2;
|
||||||
|
b++;
|
||||||
|
console.log(a + 1);
|
||||||
|
console.log(b + 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
function f1() {
|
||||||
|
var a = 1, b = 2;
|
||||||
|
--b;
|
||||||
|
console.log(a + 1);
|
||||||
|
console.log(b + 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
function f2() {
|
||||||
|
var a = 1, b = 2, c = 3;
|
||||||
|
b = c;
|
||||||
|
console.log(a + b);
|
||||||
|
console.log(b + c);
|
||||||
|
console.log(a + c);
|
||||||
|
console.log(a + b + c);
|
||||||
|
}
|
||||||
|
|
||||||
|
function f3() {
|
||||||
|
var a = 1, b = 2, c = 3;
|
||||||
|
b *= c;
|
||||||
|
console.log(a + b);
|
||||||
|
console.log(b + c);
|
||||||
|
console.log(a + c);
|
||||||
|
console.log(a + b + c);
|
||||||
|
}
|
||||||
|
|
||||||
|
function f4() {
|
||||||
|
var a = 1, b = 2, c = 3;
|
||||||
|
if (a) {
|
||||||
|
b = c;
|
||||||
|
} else {
|
||||||
|
c = b;
|
||||||
|
}
|
||||||
|
console.log(a + b);
|
||||||
|
console.log(b + c);
|
||||||
|
console.log(a + c);
|
||||||
|
console.log(a + b + c);
|
||||||
|
}
|
||||||
|
|
||||||
|
function f5(a) {
|
||||||
|
B = a;
|
||||||
|
console.log(A ? 'yes' : 'no');
|
||||||
|
console.log(B ? 'yes' : 'no');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
function f0() {
|
||||||
|
var b = 2;
|
||||||
|
b++;
|
||||||
|
console.log(2);
|
||||||
|
console.log(b + 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
function f1() {
|
||||||
|
var b = 2;
|
||||||
|
--b;
|
||||||
|
console.log(2);
|
||||||
|
console.log(b + 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
function f2() {
|
||||||
|
var b = 2;
|
||||||
|
b = 3;
|
||||||
|
console.log(1 + b);
|
||||||
|
console.log(b + 3);
|
||||||
|
console.log(4);
|
||||||
|
console.log(1 + b + 3);
|
||||||
|
}
|
||||||
|
|
||||||
|
function f3() {
|
||||||
|
var b = 2;
|
||||||
|
b *= 3;
|
||||||
|
console.log(1 + b);
|
||||||
|
console.log(b + 3);
|
||||||
|
console.log(4);
|
||||||
|
console.log(1 + b + 3);
|
||||||
|
}
|
||||||
|
|
||||||
|
function f4() {
|
||||||
|
var b = 2, c = 3;
|
||||||
|
b = c;
|
||||||
|
console.log(1 + b);
|
||||||
|
console.log(b + c);
|
||||||
|
console.log(1 + c);
|
||||||
|
console.log(1 + b + c);
|
||||||
|
}
|
||||||
|
|
||||||
|
function f5(a) {
|
||||||
|
B = a;
|
||||||
|
console.log(A ? 'yes' : 'no');
|
||||||
|
console.log(B ? 'yes' : 'no');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
unsafe_evaluate: {
|
||||||
|
options = {
|
||||||
|
evaluate : true,
|
||||||
|
reduce_vars : true,
|
||||||
|
unsafe : true,
|
||||||
|
unused : true
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
function f0(){
|
||||||
|
var a = {
|
||||||
|
b:1
|
||||||
|
};
|
||||||
|
console.log(a.b + 3);
|
||||||
|
}
|
||||||
|
|
||||||
|
function f1(){
|
||||||
|
var a = {
|
||||||
|
b:{
|
||||||
|
c:1
|
||||||
|
},
|
||||||
|
d:2
|
||||||
|
};
|
||||||
|
console.log(a.b + 3, a.d + 4, a.b.c + 5, a.d.c + 6);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
function f0(){
|
||||||
|
console.log(4);
|
||||||
|
}
|
||||||
|
|
||||||
|
function f1(){
|
||||||
|
var a = {
|
||||||
|
b:{
|
||||||
|
c:1
|
||||||
|
},
|
||||||
|
d:2
|
||||||
|
};
|
||||||
|
console.log(a.b + 3, 6, 6, 2..c + 6);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
unsafe_evaluate_object: {
|
||||||
|
options = {
|
||||||
|
evaluate : true,
|
||||||
|
reduce_vars : true,
|
||||||
|
unsafe : true
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
function f0(){
|
||||||
|
var a = 1;
|
||||||
|
var b = {};
|
||||||
|
b[a] = 2;
|
||||||
|
console.log(a + 3);
|
||||||
|
}
|
||||||
|
|
||||||
|
function f1(){
|
||||||
|
var a = {
|
||||||
|
b:1
|
||||||
|
};
|
||||||
|
a.b = 2;
|
||||||
|
console.log(a.b + 3);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
function f0(){
|
||||||
|
var a = 1;
|
||||||
|
var b = {};
|
||||||
|
b[1] = 2;
|
||||||
|
console.log(4);
|
||||||
|
}
|
||||||
|
|
||||||
|
function f1(){
|
||||||
|
var a = {
|
||||||
|
b:1
|
||||||
|
};
|
||||||
|
a.b = 2;
|
||||||
|
console.log(a.b + 3);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
unsafe_evaluate_array: {
|
||||||
|
options = {
|
||||||
|
evaluate : true,
|
||||||
|
reduce_vars : true,
|
||||||
|
unsafe : true
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
function f0(){
|
||||||
|
var a = 1;
|
||||||
|
var b = [];
|
||||||
|
b[a] = 2;
|
||||||
|
console.log(a + 3);
|
||||||
|
}
|
||||||
|
|
||||||
|
function f1(){
|
||||||
|
var a = [1];
|
||||||
|
a[2] = 3;
|
||||||
|
console.log(a.length);
|
||||||
|
}
|
||||||
|
|
||||||
|
function f2(){
|
||||||
|
var a = [1];
|
||||||
|
a.push(2);
|
||||||
|
console.log(a.length);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
function f0(){
|
||||||
|
var a = 1;
|
||||||
|
var b = [];
|
||||||
|
b[1] = 2;
|
||||||
|
console.log(4);
|
||||||
|
}
|
||||||
|
|
||||||
|
function f1(){
|
||||||
|
var a = [1];
|
||||||
|
a[2] = 3;
|
||||||
|
console.log(a.length);
|
||||||
|
}
|
||||||
|
|
||||||
|
function f2(){
|
||||||
|
var a = [1];
|
||||||
|
a.push(2);
|
||||||
|
console.log(a.length);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
unsafe_evaluate_equality: {
|
||||||
|
options = {
|
||||||
|
evaluate : true,
|
||||||
|
reduce_vars : true,
|
||||||
|
unsafe : true,
|
||||||
|
unused : true
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
function f0(){
|
||||||
|
var a = {};
|
||||||
|
console.log(a === a);
|
||||||
|
}
|
||||||
|
|
||||||
|
function f1(){
|
||||||
|
var a = [];
|
||||||
|
console.log(a === a);
|
||||||
|
}
|
||||||
|
|
||||||
|
function f2(){
|
||||||
|
var a = {a:1, b:2};
|
||||||
|
var b = a;
|
||||||
|
var c = a;
|
||||||
|
console.log(b === c);
|
||||||
|
}
|
||||||
|
|
||||||
|
function f3(){
|
||||||
|
var a = [1, 2, 3];
|
||||||
|
var b = a;
|
||||||
|
var c = a;
|
||||||
|
console.log(b === c);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
function f0(){
|
||||||
|
console.log(true);
|
||||||
|
}
|
||||||
|
|
||||||
|
function f1(){
|
||||||
|
console.log(true);
|
||||||
|
}
|
||||||
|
|
||||||
|
function f2(){
|
||||||
|
console.log(true);
|
||||||
|
}
|
||||||
|
|
||||||
|
function f3(){
|
||||||
|
console.log(true);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
passes: {
|
||||||
|
options = {
|
||||||
|
conditionals: true,
|
||||||
|
evaluate: true,
|
||||||
|
passes: 2,
|
||||||
|
reduce_vars: true,
|
||||||
|
unused: true,
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
function f() {
|
||||||
|
var a = 1, b = 2, c = 3;
|
||||||
|
if (a) {
|
||||||
|
b = c;
|
||||||
|
} else {
|
||||||
|
c = b;
|
||||||
|
}
|
||||||
|
console.log(a + b);
|
||||||
|
console.log(b + c);
|
||||||
|
console.log(a + c);
|
||||||
|
console.log(a + b + c);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
function f() {
|
||||||
|
var b = 2;
|
||||||
|
b = 3;
|
||||||
|
console.log(1 + b);
|
||||||
|
console.log(b + 3);
|
||||||
|
console.log(4);
|
||||||
|
console.log(1 + b + 3);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
iife: {
|
||||||
|
options = {
|
||||||
|
evaluate: true,
|
||||||
|
reduce_vars: true,
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
!function(a, b, c) {
|
||||||
|
b++;
|
||||||
|
console.log(a - 1, b * 1, c + 2);
|
||||||
|
}(1, 2, 3);
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
!function(a, b, c) {
|
||||||
|
b++;
|
||||||
|
console.log(0, 1 * b, 5);
|
||||||
|
}(1, 2, 3);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
iife_new: {
|
||||||
|
options = {
|
||||||
|
evaluate: true,
|
||||||
|
reduce_vars: true,
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
var A = new function(a, b, c) {
|
||||||
|
b++;
|
||||||
|
console.log(a - 1, b * 1, c + 2);
|
||||||
|
}(1, 2, 3);
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
var A = new function(a, b, c) {
|
||||||
|
b++;
|
||||||
|
console.log(0, 1 * b, 5);
|
||||||
|
}(1, 2, 3);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
multi_def: {
|
||||||
|
options = {
|
||||||
|
evaluate: true,
|
||||||
|
reduce_vars: true,
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
function f(a) {
|
||||||
|
if (a)
|
||||||
|
var b = 1;
|
||||||
|
else
|
||||||
|
var b = 2
|
||||||
|
console.log(b + 1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
function f(a) {
|
||||||
|
if (a)
|
||||||
|
var b = 1;
|
||||||
|
else
|
||||||
|
var b = 2
|
||||||
|
console.log(b + 1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
multi_def_2: {
|
||||||
|
options = {
|
||||||
|
evaluate: true,
|
||||||
|
reduce_vars: true,
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
if (code == 16)
|
||||||
|
var bitsLength = 2, bitsOffset = 3, what = len;
|
||||||
|
else if (code == 17)
|
||||||
|
var bitsLength = 3, bitsOffset = 3, what = (len = 0);
|
||||||
|
else if (code == 18)
|
||||||
|
var bitsLength = 7, bitsOffset = 11, what = (len = 0);
|
||||||
|
var repeatLength = this.getBits(bitsLength) + bitsOffset;
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
if (16 == code)
|
||||||
|
var bitsLength = 2, bitsOffset = 3, what = len;
|
||||||
|
else if (17 == code)
|
||||||
|
var bitsLength = 3, bitsOffset = 3, what = (len = 0);
|
||||||
|
else if (18 == code)
|
||||||
|
var bitsLength = 7, bitsOffset = 11, what = (len = 0);
|
||||||
|
var repeatLength = this.getBits(bitsLength) + bitsOffset;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -15,4 +15,120 @@ dont_screw: {
|
|||||||
|
|
||||||
input: f("\v");
|
input: f("\v");
|
||||||
expect_exact: 'f("\\x0B");';
|
expect_exact: 'f("\\x0B");';
|
||||||
}
|
}
|
||||||
|
|
||||||
|
do_screw_try_catch: {
|
||||||
|
options = { screw_ie8: true };
|
||||||
|
mangle = { screw_ie8: true };
|
||||||
|
beautify = { screw_ie8: true };
|
||||||
|
input: {
|
||||||
|
good = function(e){
|
||||||
|
return function(error){
|
||||||
|
try{
|
||||||
|
e()
|
||||||
|
} catch(e) {
|
||||||
|
error(e)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
good = function(n){
|
||||||
|
return function(t){
|
||||||
|
try{
|
||||||
|
n()
|
||||||
|
} catch(n) {
|
||||||
|
t(n)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
dont_screw_try_catch: {
|
||||||
|
// This test is known to generate incorrect code for screw_ie8=false.
|
||||||
|
// Update expected result in the event this bug is ever fixed.
|
||||||
|
options = { screw_ie8: false };
|
||||||
|
mangle = { screw_ie8: false };
|
||||||
|
beautify = { screw_ie8: false };
|
||||||
|
input: {
|
||||||
|
bad = function(e){
|
||||||
|
return function(error){
|
||||||
|
try{
|
||||||
|
e()
|
||||||
|
} catch(e) {
|
||||||
|
error(e)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
bad = function(n){
|
||||||
|
return function(n){
|
||||||
|
try{
|
||||||
|
t()
|
||||||
|
} catch(t) {
|
||||||
|
n(t)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
do_screw_try_catch_undefined: {
|
||||||
|
options = { screw_ie8: true };
|
||||||
|
mangle = { screw_ie8: true };
|
||||||
|
beautify = { screw_ie8: true };
|
||||||
|
input: {
|
||||||
|
function a(b){
|
||||||
|
try {
|
||||||
|
throw 'Stuff';
|
||||||
|
} catch (undefined) {
|
||||||
|
console.log('caught: ' + undefined);
|
||||||
|
}
|
||||||
|
console.log('undefined is ' + undefined);
|
||||||
|
return b === undefined;
|
||||||
|
};
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
function a(o){
|
||||||
|
try{
|
||||||
|
throw "Stuff"
|
||||||
|
} catch (o) {
|
||||||
|
console.log("caught: "+o)
|
||||||
|
}
|
||||||
|
console.log("undefined is " + void 0);
|
||||||
|
return void 0===o
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
dont_screw_try_catch_undefined: {
|
||||||
|
// This test is known to generate incorrect code for screw_ie8=false.
|
||||||
|
// Update expected result in the event this bug is ever fixed.
|
||||||
|
options = { screw_ie8: false };
|
||||||
|
mangle = { screw_ie8: false };
|
||||||
|
beautify = { screw_ie8: false };
|
||||||
|
input: {
|
||||||
|
function a(b){
|
||||||
|
try {
|
||||||
|
throw 'Stuff';
|
||||||
|
} catch (undefined) {
|
||||||
|
console.log('caught: ' + undefined);
|
||||||
|
}
|
||||||
|
console.log('undefined is ' + undefined);
|
||||||
|
return b === undefined;
|
||||||
|
};
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
function a(o){
|
||||||
|
try{
|
||||||
|
throw "Stuff"
|
||||||
|
} catch (n) {
|
||||||
|
console.log("caught: "+n)
|
||||||
|
}
|
||||||
|
console.log("undefined is " + n);
|
||||||
|
return o === n
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|||||||
@@ -169,3 +169,85 @@ for_sequences: {
|
|||||||
for (y = 5; false;);
|
for (y = 5; false;);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
limit_1: {
|
||||||
|
options = {
|
||||||
|
sequences: 3,
|
||||||
|
};
|
||||||
|
input: {
|
||||||
|
a;
|
||||||
|
b;
|
||||||
|
c;
|
||||||
|
d;
|
||||||
|
e;
|
||||||
|
f;
|
||||||
|
g;
|
||||||
|
h;
|
||||||
|
i;
|
||||||
|
j;
|
||||||
|
k;
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
a, b, c;
|
||||||
|
d, e, f;
|
||||||
|
g, h, i;
|
||||||
|
j, k;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
limit_2: {
|
||||||
|
options = {
|
||||||
|
sequences: 3,
|
||||||
|
};
|
||||||
|
input: {
|
||||||
|
a, b;
|
||||||
|
c, d;
|
||||||
|
e, f;
|
||||||
|
g, h;
|
||||||
|
i, j;
|
||||||
|
k;
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
a, b, c, d;
|
||||||
|
e, f, g, h;
|
||||||
|
i, j, k;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
negate_iife_for: {
|
||||||
|
options = {
|
||||||
|
sequences: true,
|
||||||
|
negate_iife: true,
|
||||||
|
};
|
||||||
|
input: {
|
||||||
|
(function() {})();
|
||||||
|
for (i = 0; i < 5; i++) console.log(i);
|
||||||
|
|
||||||
|
(function() {})();
|
||||||
|
for (; i < 5; i++) console.log(i);
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
for (!function() {}(), i = 0; i < 5; i++) console.log(i);
|
||||||
|
for (function() {}(); i < 5; i++) console.log(i);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
iife: {
|
||||||
|
options = {
|
||||||
|
sequences: true,
|
||||||
|
};
|
||||||
|
input: {
|
||||||
|
x = 42;
|
||||||
|
(function a() {})();
|
||||||
|
!function b() {}();
|
||||||
|
~function c() {}();
|
||||||
|
+function d() {}();
|
||||||
|
-function e() {}();
|
||||||
|
void function f() {}();
|
||||||
|
typeof function g() {}();
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
x = 42, function a() {}(), function b() {}(), function c() {}(),
|
||||||
|
function d() {}(), function e() {}(), function f() {}(), function g() {}()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|||||||
10
test/compress/string-literal.js
Normal file
10
test/compress/string-literal.js
Normal file
@@ -0,0 +1,10 @@
|
|||||||
|
octal_escape_sequence: {
|
||||||
|
input: {
|
||||||
|
var boundaries = "\0\7\00\07\70\77\000\077\300\377";
|
||||||
|
var border_check = "\400\700\0000\3000";
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
var boundaries = "\x00\x07\x00\x07\x38\x3f\x00\x3f\xc0\xff";
|
||||||
|
var border_check = "\x20\x30\x38\x30\x00\x30\xc0\x30";
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -23,3 +23,25 @@ typeof_evaluation: {
|
|||||||
h='undefined';
|
h='undefined';
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
typeof_in_boolean_context: {
|
||||||
|
options = {
|
||||||
|
booleans : true,
|
||||||
|
evaluate : true,
|
||||||
|
conditionals : true,
|
||||||
|
};
|
||||||
|
input: {
|
||||||
|
function f1(x) { return typeof x ? "yes" : "no"; }
|
||||||
|
function f2() { return typeof g()? "Yes" : "No"; }
|
||||||
|
typeof 0 ? foo() : bar();
|
||||||
|
!typeof console.log(1);
|
||||||
|
var a = !typeof console.log(2);
|
||||||
|
}
|
||||||
|
expect: {
|
||||||
|
function f1(x) { return "yes"; }
|
||||||
|
function f2() { return g(), "Yes"; }
|
||||||
|
foo();
|
||||||
|
!(console.log(1), !0);
|
||||||
|
var a = !(console.log(2), !0);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|||||||
48
test/compress/wrap_iife.js
Normal file
48
test/compress/wrap_iife.js
Normal file
@@ -0,0 +1,48 @@
|
|||||||
|
wrap_iife: {
|
||||||
|
options = {
|
||||||
|
negate_iife: false,
|
||||||
|
}
|
||||||
|
beautify = {
|
||||||
|
wrap_iife: true,
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
(function() {
|
||||||
|
return function() {
|
||||||
|
console.log('test')
|
||||||
|
};
|
||||||
|
})()();
|
||||||
|
}
|
||||||
|
expect_exact: '(function(){return function(){console.log("test")}})()();'
|
||||||
|
}
|
||||||
|
|
||||||
|
wrap_iife_in_expression: {
|
||||||
|
options = {
|
||||||
|
negate_iife: false,
|
||||||
|
}
|
||||||
|
beautify = {
|
||||||
|
wrap_iife: true,
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
foo = (function () {
|
||||||
|
return bar();
|
||||||
|
})();
|
||||||
|
}
|
||||||
|
expect_exact: 'foo=(function(){return bar()})();'
|
||||||
|
}
|
||||||
|
|
||||||
|
wrap_iife_in_return_call: {
|
||||||
|
options = {
|
||||||
|
negate_iife: false,
|
||||||
|
}
|
||||||
|
beautify = {
|
||||||
|
wrap_iife: true,
|
||||||
|
}
|
||||||
|
input: {
|
||||||
|
(function() {
|
||||||
|
return (function() {
|
||||||
|
console.log('test')
|
||||||
|
})();
|
||||||
|
})()();
|
||||||
|
}
|
||||||
|
expect_exact: '(function(){return(function(){console.log("test")})()})()();'
|
||||||
|
}
|
||||||
3
test/input/comments/filter.js
Normal file
3
test/input/comments/filter.js
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
// foo
|
||||||
|
/*@preserve*/
|
||||||
|
// bar
|
||||||
1
test/input/global_defs/nested.js
Normal file
1
test/input/global_defs/nested.js
Normal file
@@ -0,0 +1 @@
|
|||||||
|
console.log(C.V, C.D);
|
||||||
1
test/input/global_defs/simple.js
Normal file
1
test/input/global_defs/simple.js
Normal file
@@ -0,0 +1 @@
|
|||||||
|
console.log(D);
|
||||||
1
test/input/invalid/eof.js
Normal file
1
test/input/invalid/eof.js
Normal file
@@ -0,0 +1 @@
|
|||||||
|
foo, bar(
|
||||||
1
test/input/invalid/simple.js
Normal file
1
test/input/invalid/simple.js
Normal file
@@ -0,0 +1 @@
|
|||||||
|
function f(a{}
|
||||||
1
test/input/invalid/tab.js
Normal file
1
test/input/invalid/tab.js
Normal file
@@ -0,0 +1 @@
|
|||||||
|
foo( xyz, 0abc);
|
||||||
8
test/input/issue-1236/simple.js
Normal file
8
test/input/issue-1236/simple.js
Normal file
@@ -0,0 +1,8 @@
|
|||||||
|
"use strict";
|
||||||
|
|
||||||
|
var foo = function foo(x) {
|
||||||
|
return "foo " + x;
|
||||||
|
};
|
||||||
|
console.log(foo("bar"));
|
||||||
|
|
||||||
|
//# sourceMappingURL=simple.js.map
|
||||||
8
test/input/issue-1236/simple.js.map
Normal file
8
test/input/issue-1236/simple.js.map
Normal file
@@ -0,0 +1,8 @@
|
|||||||
|
{
|
||||||
|
"version": 3,
|
||||||
|
"sources": ["index.js"],
|
||||||
|
"names": [],
|
||||||
|
"mappings": ";;AAAA,IAAI,MAAM,SAAN,GAAM;AAAA,SAAK,SAAS,CAAd;AAAA,CAAV;AACA,QAAQ,GAAR,CAAY,IAAI,KAAJ,CAAZ",
|
||||||
|
"file": "simple.js",
|
||||||
|
"sourcesContent": ["let foo = x => \"foo \" + x;\nconsole.log(foo(\"bar\"));"]
|
||||||
|
}
|
||||||
4
test/input/issue-1242/bar.es5
Normal file
4
test/input/issue-1242/bar.es5
Normal file
@@ -0,0 +1,4 @@
|
|||||||
|
function bar(x) {
|
||||||
|
var triple = x * (2 + 1);
|
||||||
|
return triple;
|
||||||
|
}
|
||||||
4
test/input/issue-1242/baz.es5
Normal file
4
test/input/issue-1242/baz.es5
Normal file
@@ -0,0 +1,4 @@
|
|||||||
|
function baz(x) {
|
||||||
|
var half = x / 2;
|
||||||
|
return half;
|
||||||
|
}
|
||||||
5
test/input/issue-1242/foo.es5
Normal file
5
test/input/issue-1242/foo.es5
Normal file
@@ -0,0 +1,5 @@
|
|||||||
|
var print = console.log.bind(console);
|
||||||
|
function foo(x) {
|
||||||
|
var twice = x * 2;
|
||||||
|
print('Foo:', twice);
|
||||||
|
}
|
||||||
4
test/input/issue-1242/qux.js
Normal file
4
test/input/issue-1242/qux.js
Normal file
@@ -0,0 +1,4 @@
|
|||||||
|
var a = bar(1+2);
|
||||||
|
var b = baz(3+9);
|
||||||
|
print('q' + 'u' + 'x', a, b);
|
||||||
|
foo(5+6);
|
||||||
7
test/input/issue-1323/sample.js
Normal file
7
test/input/issue-1323/sample.js
Normal file
@@ -0,0 +1,7 @@
|
|||||||
|
var bar = (function () {
|
||||||
|
function foo (bar) {
|
||||||
|
return bar;
|
||||||
|
}
|
||||||
|
|
||||||
|
return foo;
|
||||||
|
})();
|
||||||
14
test/input/issue-1431/sample.js
Normal file
14
test/input/issue-1431/sample.js
Normal file
@@ -0,0 +1,14 @@
|
|||||||
|
function f(x) {
|
||||||
|
return function() {
|
||||||
|
function n(a) {
|
||||||
|
return a * a;
|
||||||
|
}
|
||||||
|
return x(n);
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
function g(op) {
|
||||||
|
return op(1) + op(2);
|
||||||
|
}
|
||||||
|
|
||||||
|
console.log(f(g)() == 5);
|
||||||
73
test/input/issue-1482/bracketize.js
Normal file
73
test/input/issue-1482/bracketize.js
Normal file
@@ -0,0 +1,73 @@
|
|||||||
|
if (x) {
|
||||||
|
foo();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (x) {
|
||||||
|
foo();
|
||||||
|
} else {
|
||||||
|
baz();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (x) {
|
||||||
|
foo();
|
||||||
|
} else if (y) {
|
||||||
|
bar();
|
||||||
|
} else {
|
||||||
|
baz();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (x) {
|
||||||
|
if (y) {
|
||||||
|
foo();
|
||||||
|
} else {
|
||||||
|
bar();
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
baz();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (x) {
|
||||||
|
foo();
|
||||||
|
} else if (y) {
|
||||||
|
bar();
|
||||||
|
} else if (z) {
|
||||||
|
baz();
|
||||||
|
} else {
|
||||||
|
moo();
|
||||||
|
}
|
||||||
|
|
||||||
|
function f() {
|
||||||
|
if (x) {
|
||||||
|
foo();
|
||||||
|
}
|
||||||
|
if (x) {
|
||||||
|
foo();
|
||||||
|
} else {
|
||||||
|
baz();
|
||||||
|
}
|
||||||
|
if (x) {
|
||||||
|
foo();
|
||||||
|
} else if (y) {
|
||||||
|
bar();
|
||||||
|
} else {
|
||||||
|
baz();
|
||||||
|
}
|
||||||
|
if (x) {
|
||||||
|
if (y) {
|
||||||
|
foo();
|
||||||
|
} else {
|
||||||
|
bar();
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
baz();
|
||||||
|
}
|
||||||
|
if (x) {
|
||||||
|
foo();
|
||||||
|
} else if (y) {
|
||||||
|
bar();
|
||||||
|
} else if (z) {
|
||||||
|
baz();
|
||||||
|
} else {
|
||||||
|
moo();
|
||||||
|
}
|
||||||
|
}
|
||||||
17
test/input/issue-1482/default.js
Normal file
17
test/input/issue-1482/default.js
Normal file
@@ -0,0 +1,17 @@
|
|||||||
|
if (x) foo();
|
||||||
|
|
||||||
|
if (x) foo(); else baz();
|
||||||
|
|
||||||
|
if (x) foo(); else if (y) bar(); else baz();
|
||||||
|
|
||||||
|
if (x) if (y) foo(); else bar(); else baz();
|
||||||
|
|
||||||
|
if (x) foo(); else if (y) bar(); else if (z) baz(); else moo();
|
||||||
|
|
||||||
|
function f() {
|
||||||
|
if (x) foo();
|
||||||
|
if (x) foo(); else baz();
|
||||||
|
if (x) foo(); else if (y) bar(); else baz();
|
||||||
|
if (x) if (y) foo(); else bar(); else baz();
|
||||||
|
if (x) foo(); else if (y) bar(); else if (z) baz(); else moo();
|
||||||
|
}
|
||||||
12
test/input/issue-1482/input.js
Normal file
12
test/input/issue-1482/input.js
Normal file
@@ -0,0 +1,12 @@
|
|||||||
|
if (x) foo();
|
||||||
|
if (x) foo(); else baz();
|
||||||
|
if (x) foo(); else if (y) bar(); else baz();
|
||||||
|
if (x) if (y) foo(); else bar(); else baz();
|
||||||
|
if (x) foo(); else if (y) bar(); else if (z) baz(); else moo();
|
||||||
|
function f() {
|
||||||
|
if (x) foo();
|
||||||
|
if (x) foo(); else baz();
|
||||||
|
if (x) foo(); else if (y) bar(); else baz();
|
||||||
|
if (x) if (y) foo(); else bar(); else baz();
|
||||||
|
if (x) foo(); else if (y) bar(); else if (z) baz(); else moo();
|
||||||
|
}
|
||||||
3
test/input/issue-520/input.js
Normal file
3
test/input/issue-520/input.js
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
var Foo = function Foo(){console.log(1+2);}; new Foo();
|
||||||
|
|
||||||
|
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjpudWxsLCJzb3VyY2VzIjpbInN0ZGluIl0sInNvdXJjZXNDb250ZW50IjpbImNsYXNzIEZvbyB7IGNvbnN0cnVjdG9yKCl7Y29uc29sZS5sb2coMSsyKTt9IH0gbmV3IEZvbygpO1xuIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLElBQU0sR0FBRyxHQUFDLEFBQUUsWUFBVyxFQUFFLENBQUMsT0FBTyxDQUFDLEdBQUcsQ0FBQyxDQUFDLENBQUMsQ0FBQyxDQUFDLENBQUMsQ0FBQyxDQUFBLEFBQUUsQ0FBQyxJQUFJLEdBQUcsRUFBRSxDQUFDOyJ9
|
||||||
2
test/input/issue-520/output.js
Normal file
2
test/input/issue-520/output.js
Normal file
@@ -0,0 +1,2 @@
|
|||||||
|
new function(){console.log(3)};
|
||||||
|
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbInN0ZGluIl0sIm5hbWVzIjpbImNvbnNvbGUiLCJsb2ciXSwibWFwcGluZ3MiOiJBQUErQyxHQUFyQyxZQUFnQkEsUUFBUUMsSUFBSSIsInNvdXJjZXNDb250ZW50IjpbImNsYXNzIEZvbyB7IGNvbnN0cnVjdG9yKCl7Y29uc29sZS5sb2coMSsyKTt9IH0gbmV3IEZvbygpO1xuIl19
|
||||||
87
test/jetstream.js
Normal file
87
test/jetstream.js
Normal file
@@ -0,0 +1,87 @@
|
|||||||
|
#! /usr/bin/env node
|
||||||
|
// -*- js -*-
|
||||||
|
|
||||||
|
"use strict";
|
||||||
|
|
||||||
|
var site = "http://browserbench.org/JetStream/";
|
||||||
|
if (typeof phantom == "undefined") {
|
||||||
|
// workaround for tty output truncation upon process.exit()
|
||||||
|
[process.stdout, process.stderr].forEach(function(stream){
|
||||||
|
if (stream._handle && stream._handle.setBlocking)
|
||||||
|
stream._handle.setBlocking(true);
|
||||||
|
});
|
||||||
|
var args = process.argv.slice(2);
|
||||||
|
if (!args.length) {
|
||||||
|
args.push("-mc", "warnings=false");
|
||||||
|
}
|
||||||
|
args.push("--stats");
|
||||||
|
var child_process = require("child_process");
|
||||||
|
try {
|
||||||
|
require("phantomjs-prebuilt");
|
||||||
|
} catch(e) {
|
||||||
|
child_process.execSync("npm install phantomjs-prebuilt@2.1.14");
|
||||||
|
}
|
||||||
|
var http = require("http");
|
||||||
|
var server = http.createServer(function(request, response) {
|
||||||
|
request.resume();
|
||||||
|
var url = decodeURIComponent(request.url.slice(1));
|
||||||
|
var stderr = "";
|
||||||
|
var uglifyjs = child_process.fork("bin/uglifyjs", args, {
|
||||||
|
silent: true
|
||||||
|
}).on("exit", function(code) {
|
||||||
|
console.log("uglifyjs", url.indexOf(site) == 0 ? url.slice(site.length) : url, args.join(" "));
|
||||||
|
console.log(stderr);
|
||||||
|
if (code) throw new Error("uglifyjs failed with code " + code);
|
||||||
|
});
|
||||||
|
uglifyjs.stderr.on("data", function(data) {
|
||||||
|
stderr += data;
|
||||||
|
}).setEncoding("utf8");
|
||||||
|
uglifyjs.stdout.pipe(response);
|
||||||
|
http.get(url, function(res) {
|
||||||
|
res.pipe(uglifyjs.stdin);
|
||||||
|
});
|
||||||
|
}).listen().on("listening", function() {
|
||||||
|
var phantomjs = require("phantomjs-prebuilt");
|
||||||
|
var program = phantomjs.exec(process.argv[1], server.address().port);
|
||||||
|
program.stdout.pipe(process.stdout);
|
||||||
|
program.stderr.pipe(process.stderr);
|
||||||
|
program.on("exit", function(code) {
|
||||||
|
server.close();
|
||||||
|
if (code) throw new Error("JetStream failed!");
|
||||||
|
console.log("JetStream completed successfully.");
|
||||||
|
});
|
||||||
|
});
|
||||||
|
server.timeout = 0;
|
||||||
|
} else {
|
||||||
|
var page = require("webpage").create();
|
||||||
|
page.onError = function(msg, trace) {
|
||||||
|
var body = [ msg ];
|
||||||
|
if (trace) trace.forEach(function(t) {
|
||||||
|
body.push(" " + (t.function || "Anonymous function") + " (" + t.file + ":" + t.line + ")");
|
||||||
|
});
|
||||||
|
console.error(body.join("\n"));
|
||||||
|
phantom.exit(1);
|
||||||
|
};
|
||||||
|
var url = "http://localhost:" + require("system").args[1] + "/";
|
||||||
|
page.onResourceRequested = function(requestData, networkRequest) {
|
||||||
|
if (/\.js$/.test(requestData.url))
|
||||||
|
networkRequest.changeUrl(url + encodeURIComponent(requestData.url));
|
||||||
|
}
|
||||||
|
page.onConsoleMessage = function(msg) {
|
||||||
|
if (/Error:/i.test(msg)) {
|
||||||
|
console.error(msg);
|
||||||
|
phantom.exit(1);
|
||||||
|
}
|
||||||
|
console.log(msg);
|
||||||
|
if (~msg.indexOf("Raw results:")) {
|
||||||
|
phantom.exit();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
page.open(site, function(status) {
|
||||||
|
if (status != "success") phantomjs.exit(1);
|
||||||
|
page.evaluate(function() {
|
||||||
|
JetStream.switchToQuick();
|
||||||
|
JetStream.start();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
32
test/mocha/accessorTokens-1492.js
Normal file
32
test/mocha/accessorTokens-1492.js
Normal file
@@ -0,0 +1,32 @@
|
|||||||
|
var UglifyJS = require('../../');
|
||||||
|
var assert = require("assert");
|
||||||
|
|
||||||
|
describe("Accessor tokens", function() {
|
||||||
|
it("Should fill the token information for accessors (issue #1492)", function() {
|
||||||
|
// location 0 1 2 3 4
|
||||||
|
// 01234567890123456789012345678901234567890123456789
|
||||||
|
var ast = UglifyJS.parse("var obj = { get latest() { return undefined; } }");
|
||||||
|
|
||||||
|
// test all AST_ObjectProperty tokens are set as expected
|
||||||
|
var checkedAST_ObjectProperty = false;
|
||||||
|
var checkWalker = new UglifyJS.TreeWalker(function(node, descend) {
|
||||||
|
if (node instanceof UglifyJS.AST_ObjectProperty) {
|
||||||
|
checkedAST_ObjectProperty = true;
|
||||||
|
|
||||||
|
assert.equal(node.start.pos, 12);
|
||||||
|
assert.equal(node.end.endpos, 46);
|
||||||
|
|
||||||
|
assert(node.key instanceof UglifyJS.AST_SymbolRef);
|
||||||
|
assert.equal(node.key.start.pos, 16);
|
||||||
|
assert.equal(node.key.end.endpos, 22);
|
||||||
|
|
||||||
|
assert(node.value instanceof UglifyJS.AST_Accessor);
|
||||||
|
assert.equal(node.value.start.pos, 22);
|
||||||
|
assert.equal(node.value.end.endpos, 46);
|
||||||
|
|
||||||
|
}
|
||||||
|
});
|
||||||
|
ast.walk(checkWalker);
|
||||||
|
assert(checkedAST_ObjectProperty, "AST_ObjectProperty not found");
|
||||||
|
});
|
||||||
|
});
|
||||||
@@ -19,4 +19,12 @@ describe("arguments", function() {
|
|||||||
value // Select function as scope
|
value // Select function as scope
|
||||||
);
|
);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
it("Should recognize when a function uses arguments", function() {
|
||||||
|
var ast = UglifyJS.parse("function a(){function b(){function c(){}; return arguments[0];}}");
|
||||||
|
ast.figure_out_scope();
|
||||||
|
assert.strictEqual(ast.body[0].uses_arguments, false);
|
||||||
|
assert.strictEqual(ast.body[0].body[0].uses_arguments, true);
|
||||||
|
assert.strictEqual(ast.body[0].body[0].body[0].uses_arguments, false);
|
||||||
|
});
|
||||||
});
|
});
|
||||||
241
test/mocha/cli.js
Normal file
241
test/mocha/cli.js
Normal file
@@ -0,0 +1,241 @@
|
|||||||
|
var assert = require("assert");
|
||||||
|
var exec = require("child_process").exec;
|
||||||
|
var readFileSync = require("fs").readFileSync;
|
||||||
|
|
||||||
|
describe("bin/uglifyjs", function () {
|
||||||
|
var uglifyjscmd = '"' + process.argv[0] + '" bin/uglifyjs';
|
||||||
|
it("should produce a functional build when using --self", function (done) {
|
||||||
|
this.timeout(15000);
|
||||||
|
|
||||||
|
var command = uglifyjscmd + ' --self -cm --wrap WrappedUglifyJS';
|
||||||
|
|
||||||
|
exec(command, function (err, stdout) {
|
||||||
|
if (err) throw err;
|
||||||
|
|
||||||
|
eval(stdout);
|
||||||
|
|
||||||
|
assert.strictEqual(typeof WrappedUglifyJS, 'object');
|
||||||
|
assert.strictEqual(true, WrappedUglifyJS.parse('foo;') instanceof WrappedUglifyJS.AST_Node);
|
||||||
|
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
it("Should be able to filter comments correctly with `--comment all`", function (done) {
|
||||||
|
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");
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
it("Should be able to filter comments correctly with `--comment <RegExp>`", function (done) {
|
||||||
|
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");
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
it("Should be able to filter comments correctly with just `--comment`", function (done) {
|
||||||
|
var command = uglifyjscmd + ' test/input/comments/filter.js --comments';
|
||||||
|
|
||||||
|
exec(command, function (err, stdout) {
|
||||||
|
if (err) throw err;
|
||||||
|
|
||||||
|
assert.strictEqual(stdout, "/*@preserve*/\n\n");
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
it("Should append source map to output when using --source-map-inline", function (done) {
|
||||||
|
var command = uglifyjscmd + ' test/input/issue-1323/sample.js --source-map-inline';
|
||||||
|
|
||||||
|
exec(command, function (err, stdout) {
|
||||||
|
if (err) throw err;
|
||||||
|
|
||||||
|
assert.strictEqual(stdout, "var bar=function(){function foo(bar){return bar}return foo}();\n" +
|
||||||
|
"//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbInRlc3QvaW5wdXQvaXNzdWUtMTMyMy9zYW1wbGUuanMiXSwibmFtZXMiOlsiYmFyIiwiZm9vIl0sIm1hcHBpbmdzIjoiQUFBQSxHQUFJQSxLQUFNLFdBQ04sUUFBU0MsS0FBS0QsS0FDVixNQUFPQSxLQUdYLE1BQU9DIn0=\n");
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
it("should not append source map to output when not using --source-map-inline", function (done) {
|
||||||
|
var command = uglifyjscmd + ' test/input/issue-1323/sample.js';
|
||||||
|
|
||||||
|
exec(command, function (err, stdout) {
|
||||||
|
if (err) throw err;
|
||||||
|
|
||||||
|
assert.strictEqual(stdout, "var bar=function(){function foo(bar){return bar}return foo}();\n");
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
it("Should work with --keep-fnames (mangle only)", function (done) {
|
||||||
|
var command = uglifyjscmd + ' test/input/issue-1431/sample.js --keep-fnames -m';
|
||||||
|
|
||||||
|
exec(command, function (err, stdout) {
|
||||||
|
if (err) throw err;
|
||||||
|
|
||||||
|
assert.strictEqual(stdout, "function f(r){return function(){function n(n){return n*n}return r(n)}}function g(n){return n(1)+n(2)}console.log(f(g)()==5);\n");
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
it("Should work with --keep-fnames (mangle & compress)", function (done) {
|
||||||
|
var command = uglifyjscmd + ' test/input/issue-1431/sample.js --keep-fnames -m -c';
|
||||||
|
|
||||||
|
exec(command, function (err, stdout) {
|
||||||
|
if (err) throw err;
|
||||||
|
|
||||||
|
assert.strictEqual(stdout, "function f(r){return function(){function n(n){return n*n}return r(n)}}function g(n){return n(1)+n(2)}console.log(5==f(g)());\n");
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
it("Should work with keep_fnames under mangler options", function (done) {
|
||||||
|
var command = uglifyjscmd + ' test/input/issue-1431/sample.js -m keep_fnames=true';
|
||||||
|
|
||||||
|
exec(command, function (err, stdout) {
|
||||||
|
if (err) throw err;
|
||||||
|
|
||||||
|
assert.strictEqual(stdout, "function f(r){return function(){function n(n){return n*n}return r(n)}}function g(n){return n(1)+n(2)}console.log(f(g)()==5);\n");
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
it("Should work with --define (simple)", function (done) {
|
||||||
|
var command = uglifyjscmd + ' test/input/global_defs/simple.js --define D=5 -c';
|
||||||
|
|
||||||
|
exec(command, function (err, stdout) {
|
||||||
|
if (err) throw err;
|
||||||
|
|
||||||
|
assert.strictEqual(stdout, "console.log(5);\n");
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
it("Should work with --define (nested)", function (done) {
|
||||||
|
var command = uglifyjscmd + ' test/input/global_defs/nested.js --define C.D=5,C.V=3 -c';
|
||||||
|
|
||||||
|
exec(command, function (err, stdout) {
|
||||||
|
if (err) throw err;
|
||||||
|
|
||||||
|
assert.strictEqual(stdout, "console.log(3,5);\n");
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
it("Should work with --define (AST_Node)", function (done) {
|
||||||
|
var command = uglifyjscmd + ' test/input/global_defs/simple.js --define console.log=stdout.println -c';
|
||||||
|
|
||||||
|
exec(command, function (err, stdout) {
|
||||||
|
if (err) throw err;
|
||||||
|
|
||||||
|
assert.strictEqual(stdout, "stdout.println(D);\n");
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
it("Should work with `--beautify`", function (done) {
|
||||||
|
var command = uglifyjscmd + ' test/input/issue-1482/input.js -b';
|
||||||
|
|
||||||
|
exec(command, function (err, stdout) {
|
||||||
|
if (err) throw err;
|
||||||
|
|
||||||
|
assert.strictEqual(stdout, readFileSync("test/input/issue-1482/default.js", "utf8"));
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
it("Should work with `--beautify bracketize`", function (done) {
|
||||||
|
var command = uglifyjscmd + ' test/input/issue-1482/input.js -b bracketize';
|
||||||
|
|
||||||
|
exec(command, function (err, stdout) {
|
||||||
|
if (err) throw err;
|
||||||
|
|
||||||
|
assert.strictEqual(stdout, readFileSync("test/input/issue-1482/bracketize.js", "utf8"));
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
it("Should process inline source map", function(done) {
|
||||||
|
var command = uglifyjscmd + ' test/input/issue-520/input.js -cm toplevel --in-source-map inline --source-map-inline';
|
||||||
|
|
||||||
|
exec(command, function (err, stdout) {
|
||||||
|
if (err) throw err;
|
||||||
|
|
||||||
|
assert.strictEqual(stdout, readFileSync("test/input/issue-520/output.js", "utf8"));
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
it("Should warn for missing inline source map", function(done) {
|
||||||
|
var command = uglifyjscmd + ' test/input/issue-1323/sample.js --in-source-map inline';
|
||||||
|
|
||||||
|
exec(command, function (err, stdout, stderr) {
|
||||||
|
if (err) throw err;
|
||||||
|
|
||||||
|
assert.strictEqual(stdout, "var bar=function(){function foo(bar){return bar}return foo}();\n");
|
||||||
|
assert.strictEqual(stderr, "WARN: inline source map not found\n");
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
it("Should fail with multiple input and inline source map", function(done) {
|
||||||
|
var command = uglifyjscmd + ' test/input/issue-520/input.js test/input/issue-520/output.js --in-source-map inline --source-map-inline';
|
||||||
|
|
||||||
|
exec(command, function (err, stdout, stderr) {
|
||||||
|
assert.ok(err);
|
||||||
|
assert.strictEqual(stderr, "ERROR: Inline source map only works with singular input\n");
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
it("Should fail with acorn and inline source map", function(done) {
|
||||||
|
var command = uglifyjscmd + ' test/input/issue-520/input.js --in-source-map inline --source-map-inline --acorn';
|
||||||
|
|
||||||
|
exec(command, function (err, stdout, stderr) {
|
||||||
|
assert.ok(err);
|
||||||
|
assert.strictEqual(stderr, "ERROR: Inline source map only works with built-in parser\n");
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
it("Should fail with SpiderMonkey and inline source map", function(done) {
|
||||||
|
var command = uglifyjscmd + ' test/input/issue-520/input.js --in-source-map inline --source-map-inline --spidermonkey';
|
||||||
|
|
||||||
|
exec(command, function (err, stdout, stderr) {
|
||||||
|
assert.ok(err);
|
||||||
|
assert.strictEqual(stderr, "ERROR: Inline source map only works with built-in parser\n");
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
it("Should fail with invalid syntax", function(done) {
|
||||||
|
var command = uglifyjscmd + ' test/input/invalid/simple.js';
|
||||||
|
|
||||||
|
exec(command, function (err, stdout, stderr) {
|
||||||
|
assert.ok(err);
|
||||||
|
var lines = stderr.split(/\n/);
|
||||||
|
assert.strictEqual(lines[0], "Parse error at test/input/invalid/simple.js:1,12");
|
||||||
|
assert.strictEqual(lines[1], "function f(a{}");
|
||||||
|
assert.strictEqual(lines[2], " ^");
|
||||||
|
assert.strictEqual(lines[3], "SyntaxError: Unexpected token punc «{», expected punc «,»");
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
it("Should fail with correct marking of tabs", function(done) {
|
||||||
|
var command = uglifyjscmd + ' test/input/invalid/tab.js';
|
||||||
|
|
||||||
|
exec(command, function (err, stdout, stderr) {
|
||||||
|
assert.ok(err);
|
||||||
|
var lines = stderr.split(/\n/);
|
||||||
|
assert.strictEqual(lines[0], "Parse error at test/input/invalid/tab.js:1,12");
|
||||||
|
assert.strictEqual(lines[1], "\t\tfoo(\txyz, 0abc);");
|
||||||
|
assert.strictEqual(lines[2], "\t\t \t ^");
|
||||||
|
assert.strictEqual(lines[3], "SyntaxError: Invalid syntax: 0abc");
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
it("Should fail with correct marking at start of line", function(done) {
|
||||||
|
var command = uglifyjscmd + ' test/input/invalid/eof.js';
|
||||||
|
|
||||||
|
exec(command, function (err, stdout, stderr) {
|
||||||
|
assert.ok(err);
|
||||||
|
var lines = stderr.split(/\n/);
|
||||||
|
assert.strictEqual(lines[0], "Parse error at test/input/invalid/eof.js:2,0");
|
||||||
|
assert.strictEqual(lines[1], "foo, bar(");
|
||||||
|
assert.strictEqual(lines[2], " ^");
|
||||||
|
assert.strictEqual(lines[3], "SyntaxError: Unexpected token: eof (undefined)");
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
@@ -2,11 +2,21 @@ var UglifyJS = require('../../');
|
|||||||
var assert = require("assert");
|
var assert = require("assert");
|
||||||
|
|
||||||
describe("comment filters", function() {
|
describe("comment filters", function() {
|
||||||
it("Should be able to filter comments by passing regex", 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");
|
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\n");
|
||||||
});
|
});
|
||||||
|
|
||||||
|
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");
|
||||||
|
});
|
||||||
|
|
||||||
|
it("Should be able to filter commments with the 'some' option", function() {
|
||||||
|
var ast = UglifyJS.parse("// foo\n/*@preserve*/\n// bar\n/*@license*/\n//@license with the wrong comment type\n/*@cc_on something*/");
|
||||||
|
assert.strictEqual(ast.print_to_string({comments: "some"}), "/*@preserve*/\n/*@license*/\n/*@cc_on something*/\n");
|
||||||
|
});
|
||||||
|
|
||||||
it("Should be able to filter comments by passing a function", function() {
|
it("Should be able to filter comments by passing a function", function() {
|
||||||
var ast = UglifyJS.parse("/*TEST 123*/\n//An other comment\n//8 chars.");
|
var ast = UglifyJS.parse("/*TEST 123*/\n//An other comment\n//8 chars.");
|
||||||
var f = function(node, comment) {
|
var f = function(node, comment) {
|
||||||
@@ -16,6 +26,11 @@ describe("comment filters", function() {
|
|||||||
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.\n");
|
||||||
});
|
});
|
||||||
|
|
||||||
|
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");
|
||||||
|
});
|
||||||
|
|
||||||
it("Should be able to get the comment and comment type when using a function", function() {
|
it("Should be able to get the comment and comment type when using a function", function() {
|
||||||
var ast = UglifyJS.parse("/*!test1*/\n/*test2*/\n//!test3\n//test4\n<!--test5\n<!--!test6\n-->test7\n-->!test8");
|
var ast = UglifyJS.parse("/*!test1*/\n/*test2*/\n//!test3\n//test4\n<!--test5\n<!--!test6\n-->test7\n-->!test8");
|
||||||
var f = function(node, comment) {
|
var f = function(node, comment) {
|
||||||
@@ -42,4 +57,19 @@ describe("comment filters", function() {
|
|||||||
|
|
||||||
assert.strictEqual(ast.print_to_string({comments: f}), "#!Random comment\n//test1\n/*test2*/\n");
|
assert.strictEqual(ast.print_to_string({comments: f}), "#!Random comment\n//test1\n/*test2*/\n");
|
||||||
});
|
});
|
||||||
|
|
||||||
|
it("Should never be able to filter comment5 when using 'some' as filter", function() {
|
||||||
|
var ast = UglifyJS.parse("#!foo\n//foo\n/*@preserve*/\n/* please hide me */");
|
||||||
|
assert.strictEqual(ast.print_to_string({comments: "some"}), "#!foo\n/*@preserve*/\n");
|
||||||
|
});
|
||||||
|
|
||||||
|
it("Should have no problem on multiple calls", function() {
|
||||||
|
const options = {
|
||||||
|
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(){}");
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|||||||
50
test/mocha/comment.js
Normal file
50
test/mocha/comment.js
Normal file
@@ -0,0 +1,50 @@
|
|||||||
|
var assert = require("assert");
|
||||||
|
var uglify = require("../../");
|
||||||
|
|
||||||
|
describe("Comment", function() {
|
||||||
|
it("Should recognize eol of single line comments", function() {
|
||||||
|
var tests = [
|
||||||
|
"//Some comment 1\n>",
|
||||||
|
"//Some comment 2\r>",
|
||||||
|
"//Some comment 3\r\n>",
|
||||||
|
"//Some comment 4\u2028>",
|
||||||
|
"//Some comment 5\u2029>"
|
||||||
|
];
|
||||||
|
|
||||||
|
var fail = function(e) {
|
||||||
|
return e instanceof uglify.JS_Parse_Error &&
|
||||||
|
e.message === "Unexpected token: operator (>)" &&
|
||||||
|
e.line === 2 &&
|
||||||
|
e.col === 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (var i = 0; i < tests.length; i++) {
|
||||||
|
assert.throws(function() {
|
||||||
|
uglify.parse(tests[i], {fromString: true})
|
||||||
|
}, fail, tests[i]);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
it("Should update the position of a multiline comment correctly", function() {
|
||||||
|
var tests = [
|
||||||
|
"/*Some comment 1\n\n\n*/\n>\n\n\n\n\n\n",
|
||||||
|
"/*Some comment 2\r\n\r\n\r\n*/\r\n>\n\n\n\n\n\n",
|
||||||
|
"/*Some comment 3\r\r\r*/\r>\n\n\n\n\n\n",
|
||||||
|
"/*Some comment 4\u2028\u2028\u2028*/\u2028>\n\n\n\n\n\n",
|
||||||
|
"/*Some comment 5\u2029\u2029\u2029*/\u2029>\n\n\n\n\n\n"
|
||||||
|
];
|
||||||
|
|
||||||
|
var fail = function(e) {
|
||||||
|
return e instanceof uglify.JS_Parse_Error &&
|
||||||
|
e.message === "Unexpected token: operator (>)" &&
|
||||||
|
e.line === 5 &&
|
||||||
|
e.col === 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (var i = 0; i < tests.length; i++) {
|
||||||
|
assert.throws(function() {
|
||||||
|
uglify.parse(tests[i], {fromString: true})
|
||||||
|
}, fail, tests[i]);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
@@ -7,7 +7,7 @@ describe("comment before constant", function() {
|
|||||||
it("Should test comment before constant is retained and output after mangle.", function() {
|
it("Should test comment before constant is retained and output after mangle.", function() {
|
||||||
var result = Uglify.minify(js, {
|
var result = Uglify.minify(js, {
|
||||||
fromString: true,
|
fromString: true,
|
||||||
compress: { collapse_vars: false },
|
compress: { collapse_vars: false, reduce_vars: false },
|
||||||
mangle: {},
|
mangle: {},
|
||||||
output: { comments: true },
|
output: { comments: true },
|
||||||
});
|
});
|
||||||
@@ -17,9 +17,9 @@ describe("comment before constant", function() {
|
|||||||
it("Should test code works when comments disabled.", function() {
|
it("Should test code works when comments disabled.", function() {
|
||||||
var result = Uglify.minify(js, {
|
var result = Uglify.minify(js, {
|
||||||
fromString: true,
|
fromString: true,
|
||||||
compress: { collapse_vars: false },
|
compress: { collapse_vars: false, reduce_vars: false },
|
||||||
mangle: {},
|
mangle: {},
|
||||||
output: {},
|
output: { comments: false },
|
||||||
});
|
});
|
||||||
assert.strictEqual(result.code, 'function f(){var n=!1;return n}');
|
assert.strictEqual(result.code, 'function f(){var n=!1;return n}');
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -168,7 +168,7 @@ describe("Directives", function() {
|
|||||||
throw new Error("Expected parser to fail");
|
throw new Error("Expected parser to fail");
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
assert.strictEqual(e instanceof uglify.JS_Parse_Error, true);
|
assert.strictEqual(e instanceof uglify.JS_Parse_Error, true);
|
||||||
assert.strictEqual(e.message, "SyntaxError: Unexpected token: punc (])");
|
assert.strictEqual(e.message, "Unexpected token: punc (])");
|
||||||
}
|
}
|
||||||
|
|
||||||
test_directive(tokenizer, tests[i]);
|
test_directive(tokenizer, tests[i]);
|
||||||
|
|||||||
@@ -71,7 +71,7 @@ describe("Getters and setters", function() {
|
|||||||
var fail = function(data) {
|
var fail = function(data) {
|
||||||
return function (e) {
|
return function (e) {
|
||||||
return e instanceof UglifyJS.JS_Parse_Error &&
|
return e instanceof UglifyJS.JS_Parse_Error &&
|
||||||
e.message === "SyntaxError: Invalid getter/setter name: " + data.operator;
|
e.message === "Invalid getter/setter name: " + data.operator;
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|||||||
28
test/mocha/glob.js
Normal file
28
test/mocha/glob.js
Normal file
@@ -0,0 +1,28 @@
|
|||||||
|
var Uglify = require('../../');
|
||||||
|
var assert = require("assert");
|
||||||
|
|
||||||
|
describe("minify() with input file globs", function() {
|
||||||
|
it("minify() with one input file glob string.", function() {
|
||||||
|
var result = Uglify.minify("test/input/issue-1242/foo.*", {
|
||||||
|
compress: { collapse_vars: true }
|
||||||
|
});
|
||||||
|
assert.strictEqual(result.code, 'function foo(o){print("Foo:",2*o)}var print=console.log.bind(console);');
|
||||||
|
});
|
||||||
|
it("minify() with an array of one input file glob.", function() {
|
||||||
|
var result = Uglify.minify([
|
||||||
|
"test/input/issue-1242/b*.es5",
|
||||||
|
], {
|
||||||
|
compress: { collapse_vars: true }
|
||||||
|
});
|
||||||
|
assert.strictEqual(result.code, 'function bar(n){return 3*n}function baz(n){return n/2}');
|
||||||
|
});
|
||||||
|
it("minify() with an array of multiple input file globs.", function() {
|
||||||
|
var result = Uglify.minify([
|
||||||
|
"test/input/issue-1242/???.es5",
|
||||||
|
"test/input/issue-1242/*.js",
|
||||||
|
], {
|
||||||
|
compress: { collapse_vars: true }
|
||||||
|
});
|
||||||
|
assert.strictEqual(result.code, 'function bar(n){return 3*n}function baz(n){return n/2}function foo(n){print("Foo:",2*n)}var print=console.log.bind(console);print("qux",bar(3),baz(12)),foo(11);');
|
||||||
|
});
|
||||||
|
});
|
||||||
66
test/mocha/input-sourcemaps.js
Normal file
66
test/mocha/input-sourcemaps.js
Normal file
@@ -0,0 +1,66 @@
|
|||||||
|
var Uglify = require('../../');
|
||||||
|
var assert = require("assert");
|
||||||
|
var SourceMapConsumer = require("source-map").SourceMapConsumer;
|
||||||
|
|
||||||
|
describe("input sourcemaps", function() {
|
||||||
|
var transpilemap, map;
|
||||||
|
|
||||||
|
function getMap() {
|
||||||
|
return {
|
||||||
|
"version": 3,
|
||||||
|
"sources": ["index.js"],
|
||||||
|
"names": [],
|
||||||
|
"mappings": ";;AAAA,IAAI,MAAM,SAAN,GAAM;AAAA,SAAK,SAAS,CAAd;AAAA,CAAV;AACA,QAAQ,GAAR,CAAY,IAAI,KAAJ,CAAZ",
|
||||||
|
"file": "bundle.js",
|
||||||
|
"sourcesContent": ["let foo = x => \"foo \" + x;\nconsole.log(foo(\"bar\"));"]
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
function prepareMap(sourceMap) {
|
||||||
|
var transpiled = '"use strict";\n\n' +
|
||||||
|
'var foo = function foo(x) {\n return "foo " + x;\n};\n' +
|
||||||
|
'console.log(foo("bar"));\n\n' +
|
||||||
|
'//# sourceMappingURL=bundle.js.map';
|
||||||
|
|
||||||
|
transpilemap = sourceMap || getMap();
|
||||||
|
|
||||||
|
var result = Uglify.minify(transpiled, {
|
||||||
|
fromString: true,
|
||||||
|
inSourceMap: transpilemap,
|
||||||
|
outSourceMap: true
|
||||||
|
});
|
||||||
|
|
||||||
|
map = new SourceMapConsumer(result.map);
|
||||||
|
}
|
||||||
|
|
||||||
|
beforeEach(function () {
|
||||||
|
prepareMap();
|
||||||
|
});
|
||||||
|
|
||||||
|
it("Should copy over original sourcesContent", function() {
|
||||||
|
assert.equal(map.sourceContentFor("index.js"), transpilemap.sourcesContent[0]);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("Should copy sourcesContent if sources are relative", function () {
|
||||||
|
var relativeMap = getMap();
|
||||||
|
relativeMap.sources = ['./index.js'];
|
||||||
|
|
||||||
|
prepareMap(relativeMap);
|
||||||
|
assert.notEqual(map.sourcesContent, null);
|
||||||
|
assert.equal(map.sourcesContent.length, 1);
|
||||||
|
assert.equal(map.sourceContentFor("index.js"), transpilemap.sourcesContent[0]);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("Final sourcemap should not have invalid mappings from inputSourceMap (issue #882)", function() {
|
||||||
|
// The original source has only 2 lines, check that mappings don't have more lines
|
||||||
|
|
||||||
|
var msg = "Mapping should not have higher line number than the original file had";
|
||||||
|
map.eachMapping(function(mapping) {
|
||||||
|
assert.ok(mapping.originalLine <= 2, msg)
|
||||||
|
});
|
||||||
|
|
||||||
|
map.allGeneratedPositionsFor({source: "index.js", line: 1, column: 1}).forEach(function(pos) {
|
||||||
|
assert.ok(pos.line <= 2, msg);
|
||||||
|
})
|
||||||
|
});
|
||||||
|
});
|
||||||
@@ -37,6 +37,10 @@ describe("line-endings", function() {
|
|||||||
"/\r/",
|
"/\r/",
|
||||||
"/\u2028/",
|
"/\u2028/",
|
||||||
"/\u2029/",
|
"/\u2029/",
|
||||||
|
"/\\\n/",
|
||||||
|
"/\\\r/",
|
||||||
|
"/\\\u2028/",
|
||||||
|
"/\\\u2029/",
|
||||||
"/someRandomTextLike[]()*AndThen\n/"
|
"/someRandomTextLike[]()*AndThen\n/"
|
||||||
]
|
]
|
||||||
var test = function(input) {
|
var test = function(input) {
|
||||||
|
|||||||
48
test/mocha/minify-file-map.js
Normal file
48
test/mocha/minify-file-map.js
Normal file
@@ -0,0 +1,48 @@
|
|||||||
|
var Uglify = require('../../');
|
||||||
|
var assert = require("assert");
|
||||||
|
|
||||||
|
describe("Input file as map", function() {
|
||||||
|
it("Should accept object", function() {
|
||||||
|
var jsMap = {
|
||||||
|
'/scripts/foo.js': 'var foo = {"x": 1, y: 2, \'z\': 3};'
|
||||||
|
};
|
||||||
|
var result = Uglify.minify(jsMap, {fromString: true, outSourceMap: true});
|
||||||
|
|
||||||
|
var map = JSON.parse(result.map);
|
||||||
|
assert.strictEqual(result.code, 'var foo={x:1,y:2,z:3};');
|
||||||
|
assert.deepEqual(map.sources, ['/scripts/foo.js']);
|
||||||
|
assert.strictEqual(map.file, undefined);
|
||||||
|
|
||||||
|
result = Uglify.minify(jsMap, {fromString: true, outFileName: 'out.js'});
|
||||||
|
assert.strictEqual(result.map, null);
|
||||||
|
|
||||||
|
result = Uglify.minify(jsMap, {fromString: true, outFileName: 'out.js', outSourceMap: true});
|
||||||
|
map = JSON.parse(result.map);
|
||||||
|
assert.strictEqual(map.file, 'out.js');
|
||||||
|
});
|
||||||
|
|
||||||
|
it("Should accept array of objects and strings", function() {
|
||||||
|
var jsSeq = [
|
||||||
|
{'/scripts/foo.js': 'var foo = {"x": 1, y: 2, \'z\': 3};'},
|
||||||
|
'var bar = 15;'
|
||||||
|
];
|
||||||
|
var result = Uglify.minify(jsSeq, {fromString: true, outSourceMap: true});
|
||||||
|
|
||||||
|
var map = JSON.parse(result.map);
|
||||||
|
assert.strictEqual(result.code, 'var foo={x:1,y:2,z:3},bar=15;');
|
||||||
|
assert.strictEqual(map.sources[0], '/scripts/foo.js');
|
||||||
|
});
|
||||||
|
|
||||||
|
it("Should correctly include source", function() {
|
||||||
|
var jsSeq = [
|
||||||
|
{'/scripts/foo.js': 'var foo = {"x": 1, y: 2, \'z\': 3};'},
|
||||||
|
'var bar = 15;'
|
||||||
|
];
|
||||||
|
var result = Uglify.minify(jsSeq, {fromString: true, outSourceMap: true, sourceMapIncludeSources: true});
|
||||||
|
|
||||||
|
var map = JSON.parse(result.map);
|
||||||
|
assert.strictEqual(result.code, 'var foo={x:1,y:2,z:3},bar=15;');
|
||||||
|
assert.deepEqual(map.sourcesContent, ['var foo = {"x": 1, y: 2, \'z\': 3};', 'var bar = 15;']);
|
||||||
|
});
|
||||||
|
|
||||||
|
});
|
||||||
@@ -1,5 +1,6 @@
|
|||||||
var Uglify = require('../../');
|
var Uglify = require('../../');
|
||||||
var assert = require("assert");
|
var assert = require("assert");
|
||||||
|
var readFileSync = require("fs").readFileSync;
|
||||||
|
|
||||||
describe("minify", function() {
|
describe("minify", function() {
|
||||||
it("Should test basic sanity of minify with default options", function() {
|
it("Should test basic sanity of minify with default options", function() {
|
||||||
@@ -59,4 +60,115 @@ describe("minify", function() {
|
|||||||
'a["foo"]="bar",a.a="red",x={"bar":10};');
|
'a["foo"]="bar",a.a="red",x={"bar":10};');
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
describe("inSourceMap", function() {
|
||||||
|
it("Should read the given string filename correctly when sourceMapIncludeSources is enabled (#1236)", function() {
|
||||||
|
var result = Uglify.minify('./test/input/issue-1236/simple.js', {
|
||||||
|
outSourceMap: "simple.min.js.map",
|
||||||
|
inSourceMap: "./test/input/issue-1236/simple.js.map",
|
||||||
|
sourceMapIncludeSources: true
|
||||||
|
});
|
||||||
|
|
||||||
|
var map = JSON.parse(result.map);
|
||||||
|
|
||||||
|
assert.equal(map.file, 'simple.min.js');
|
||||||
|
assert.equal(map.sourcesContent.length, 1);
|
||||||
|
assert.equal(map.sourcesContent[0],
|
||||||
|
'let foo = x => "foo " + x;\nconsole.log(foo("bar"));');
|
||||||
|
});
|
||||||
|
it("Should process inline source map", function() {
|
||||||
|
var code = Uglify.minify("./test/input/issue-520/input.js", {
|
||||||
|
inSourceMap: "inline",
|
||||||
|
sourceMapInline: true
|
||||||
|
}).code + "\n";
|
||||||
|
assert.strictEqual(code, readFileSync("test/input/issue-520/output.js", "utf8"));
|
||||||
|
});
|
||||||
|
it("Should warn for missing inline source map", function() {
|
||||||
|
var warn_function = Uglify.AST_Node.warn_function;
|
||||||
|
var warnings = [];
|
||||||
|
Uglify.AST_Node.warn_function = function(txt) {
|
||||||
|
warnings.push(txt);
|
||||||
|
};
|
||||||
|
try {
|
||||||
|
var result = Uglify.minify("./test/input/issue-1323/sample.js", {
|
||||||
|
inSourceMap: "inline",
|
||||||
|
mangle: false,
|
||||||
|
});
|
||||||
|
assert.strictEqual(result.code, "var bar=function(){function foo(bar){return bar}return foo}();");
|
||||||
|
assert.strictEqual(warnings.length, 1);
|
||||||
|
assert.strictEqual(warnings[0], "inline source map not found");
|
||||||
|
} finally {
|
||||||
|
Uglify.AST_Node.warn_function = warn_function;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
it("Should fail with multiple input and inline source map", function() {
|
||||||
|
assert.throws(function() {
|
||||||
|
Uglify.minify([
|
||||||
|
"./test/input/issue-520/input.js",
|
||||||
|
"./test/input/issue-520/output.js"
|
||||||
|
], {
|
||||||
|
inSourceMap: "inline",
|
||||||
|
sourceMapInline: true
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
it("Should fail with SpiderMonkey and inline source map", function() {
|
||||||
|
assert.throws(function() {
|
||||||
|
Uglify.minify("./test/input/issue-520/input.js", {
|
||||||
|
inSourceMap: "inline",
|
||||||
|
sourceMapInline: true,
|
||||||
|
spidermonkey: true
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("sourceMapInline", function() {
|
||||||
|
it("should append source map to output js when sourceMapInline is enabled", function() {
|
||||||
|
var result = Uglify.minify('var a = function(foo) { return foo; };', {
|
||||||
|
fromString: true,
|
||||||
|
sourceMapInline: true
|
||||||
|
});
|
||||||
|
var code = result.code;
|
||||||
|
assert.strictEqual(code, "var a=function(n){return n};\n" +
|
||||||
|
"//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIj8iXSwibmFtZXMiOlsiYSIsImZvbyJdLCJtYXBwaW5ncyI6IkFBQUEsR0FBSUEsR0FBSSxTQUFTQyxHQUFPLE1BQU9BIn0=");
|
||||||
|
});
|
||||||
|
it("should not append source map to output js when sourceMapInline is not enabled", function() {
|
||||||
|
var result = Uglify.minify('var a = function(foo) { return foo; };', {
|
||||||
|
fromString: true
|
||||||
|
});
|
||||||
|
var code = result.code;
|
||||||
|
assert.strictEqual(code, "var a=function(n){return n};");
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("#__PURE__", function() {
|
||||||
|
it("should drop #__PURE__ hint after use", function() {
|
||||||
|
var result = Uglify.minify('//@__PURE__ comment1 #__PURE__ comment2\n foo(), bar();', {
|
||||||
|
fromString: true,
|
||||||
|
output: {
|
||||||
|
comments: "all",
|
||||||
|
beautify: false,
|
||||||
|
}
|
||||||
|
});
|
||||||
|
var code = result.code;
|
||||||
|
assert.strictEqual(code, "// comment1 comment2\nbar();");
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("JS_Parse_Error", function() {
|
||||||
|
it("should throw syntax error", function() {
|
||||||
|
assert.throws(function() {
|
||||||
|
Uglify.minify("function f(a{}", { fromString: true });
|
||||||
|
}, function(err) {
|
||||||
|
assert.ok(err instanceof Error);
|
||||||
|
assert.strictEqual(err.stack.split(/\n/)[0], "SyntaxError: Unexpected token punc «{», expected punc «,»");
|
||||||
|
assert.strictEqual(err.filename, 0);
|
||||||
|
assert.strictEqual(err.line, 1);
|
||||||
|
assert.strictEqual(err.col, 12);
|
||||||
|
return true;
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
});
|
});
|
||||||
|
|||||||
24
test/mocha/number-literal.js
Normal file
24
test/mocha/number-literal.js
Normal file
@@ -0,0 +1,24 @@
|
|||||||
|
var assert = require("assert");
|
||||||
|
var uglify = require("../../");
|
||||||
|
|
||||||
|
describe("Number literals", function () {
|
||||||
|
it("Should not allow legacy octal literals in strict mode", function() {
|
||||||
|
var inputs = [
|
||||||
|
'"use strict";00;',
|
||||||
|
'"use strict"; var foo = 00;'
|
||||||
|
];
|
||||||
|
|
||||||
|
var test = function(input) {
|
||||||
|
return function() {
|
||||||
|
uglify.parse(input);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
var error = function(e) {
|
||||||
|
return e instanceof uglify.JS_Parse_Error &&
|
||||||
|
e.message === "Legacy octal literals are not allowed in strict mode";
|
||||||
|
}
|
||||||
|
for (var i = 0; i < inputs.length; i++) {
|
||||||
|
assert.throws(test(inputs[i]), error, inputs[i]);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
489
test/mocha/operator.js
Normal file
489
test/mocha/operator.js
Normal file
@@ -0,0 +1,489 @@
|
|||||||
|
var UglifyJS = require("../../");
|
||||||
|
var assert = require("assert");
|
||||||
|
|
||||||
|
describe("operator", function() {
|
||||||
|
it("Should handle mixing of ++/+/--/- correctly", function() {
|
||||||
|
function evaluate(exp) {
|
||||||
|
return new Function("var a=1,b=2,c=" + exp + ";return{a:a,b:b,c:c}")();
|
||||||
|
}
|
||||||
|
|
||||||
|
[ "", "+", "-" ].forEach(function(p) {
|
||||||
|
[ "++a", "--a", "a", "a--", "a++" ].forEach(function(a) {
|
||||||
|
[ "+", "-" ].forEach(function(o) {
|
||||||
|
[ "", "+", "-" ].forEach(function(q) {
|
||||||
|
[ "++b", "--b", "b", "b--", "b++" ].forEach(function(b) {
|
||||||
|
var exp = [p, a, o, q, b].join(" ");
|
||||||
|
var orig = evaluate(exp);
|
||||||
|
var uglify = evaluate(UglifyJS.parse(exp).print_to_string());
|
||||||
|
assert.strictEqual(orig.a, uglify.a);
|
||||||
|
assert.strictEqual(orig.b, uglify.b);
|
||||||
|
assert.strictEqual(orig.c, uglify.c);
|
||||||
|
var beautify = evaluate(UglifyJS.parse(exp).print_to_string({
|
||||||
|
beautify: true
|
||||||
|
}));
|
||||||
|
assert.strictEqual(orig.a, beautify.a);
|
||||||
|
assert.strictEqual(orig.b, beautify.b);
|
||||||
|
assert.strictEqual(orig.c, beautify.c);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
it("Should remove extraneous spaces", function() {
|
||||||
|
[
|
||||||
|
[ "++a + ++b", "++a+ ++b" ],
|
||||||
|
[ "++a + --b", "++a+--b" ],
|
||||||
|
[ "++a + b", "++a+b" ],
|
||||||
|
[ "++a + b--", "++a+b--" ],
|
||||||
|
[ "++a + b++", "++a+b++" ],
|
||||||
|
[ "++a + + ++b", "++a+ + ++b" ],
|
||||||
|
[ "++a + + --b", "++a+ +--b" ],
|
||||||
|
[ "++a + + b", "++a+ +b" ],
|
||||||
|
[ "++a + + b--", "++a+ +b--" ],
|
||||||
|
[ "++a + + b++", "++a+ +b++" ],
|
||||||
|
[ "++a + - ++b", "++a+-++b" ],
|
||||||
|
[ "++a + - --b", "++a+- --b" ],
|
||||||
|
[ "++a + - b", "++a+-b" ],
|
||||||
|
[ "++a + - b--", "++a+-b--" ],
|
||||||
|
[ "++a + - b++", "++a+-b++" ],
|
||||||
|
[ "++a - ++b", "++a-++b" ],
|
||||||
|
[ "++a - --b", "++a- --b" ],
|
||||||
|
[ "++a - b", "++a-b" ],
|
||||||
|
[ "++a - b--", "++a-b--" ],
|
||||||
|
[ "++a - b++", "++a-b++" ],
|
||||||
|
[ "++a - + ++b", "++a-+ ++b" ],
|
||||||
|
[ "++a - + --b", "++a-+--b" ],
|
||||||
|
[ "++a - + b", "++a-+b" ],
|
||||||
|
[ "++a - + b--", "++a-+b--" ],
|
||||||
|
[ "++a - + b++", "++a-+b++" ],
|
||||||
|
[ "++a - - ++b", "++a- -++b" ],
|
||||||
|
[ "++a - - --b", "++a- - --b" ],
|
||||||
|
[ "++a - - b", "++a- -b" ],
|
||||||
|
[ "++a - - b--", "++a- -b--" ],
|
||||||
|
[ "++a - - b++", "++a- -b++" ],
|
||||||
|
[ "--a + ++b", "--a+ ++b" ],
|
||||||
|
[ "--a + --b", "--a+--b" ],
|
||||||
|
[ "--a + b", "--a+b" ],
|
||||||
|
[ "--a + b--", "--a+b--" ],
|
||||||
|
[ "--a + b++", "--a+b++" ],
|
||||||
|
[ "--a + + ++b", "--a+ + ++b" ],
|
||||||
|
[ "--a + + --b", "--a+ +--b" ],
|
||||||
|
[ "--a + + b", "--a+ +b" ],
|
||||||
|
[ "--a + + b--", "--a+ +b--" ],
|
||||||
|
[ "--a + + b++", "--a+ +b++" ],
|
||||||
|
[ "--a + - ++b", "--a+-++b" ],
|
||||||
|
[ "--a + - --b", "--a+- --b" ],
|
||||||
|
[ "--a + - b", "--a+-b" ],
|
||||||
|
[ "--a + - b--", "--a+-b--" ],
|
||||||
|
[ "--a + - b++", "--a+-b++" ],
|
||||||
|
[ "--a - ++b", "--a-++b" ],
|
||||||
|
[ "--a - --b", "--a- --b" ],
|
||||||
|
[ "--a - b", "--a-b" ],
|
||||||
|
[ "--a - b--", "--a-b--" ],
|
||||||
|
[ "--a - b++", "--a-b++" ],
|
||||||
|
[ "--a - + ++b", "--a-+ ++b" ],
|
||||||
|
[ "--a - + --b", "--a-+--b" ],
|
||||||
|
[ "--a - + b", "--a-+b" ],
|
||||||
|
[ "--a - + b--", "--a-+b--" ],
|
||||||
|
[ "--a - + b++", "--a-+b++" ],
|
||||||
|
[ "--a - - ++b", "--a- -++b" ],
|
||||||
|
[ "--a - - --b", "--a- - --b" ],
|
||||||
|
[ "--a - - b", "--a- -b" ],
|
||||||
|
[ "--a - - b--", "--a- -b--" ],
|
||||||
|
[ "--a - - b++", "--a- -b++" ],
|
||||||
|
[ "a + ++b", "a+ ++b" ],
|
||||||
|
[ "a + --b", "a+--b" ],
|
||||||
|
[ "a + b", "a+b" ],
|
||||||
|
[ "a + b--", "a+b--" ],
|
||||||
|
[ "a + b++", "a+b++" ],
|
||||||
|
[ "a + + ++b", "a+ + ++b" ],
|
||||||
|
[ "a + + --b", "a+ +--b" ],
|
||||||
|
[ "a + + b", "a+ +b" ],
|
||||||
|
[ "a + + b--", "a+ +b--" ],
|
||||||
|
[ "a + + b++", "a+ +b++" ],
|
||||||
|
[ "a + - ++b", "a+-++b" ],
|
||||||
|
[ "a + - --b", "a+- --b" ],
|
||||||
|
[ "a + - b", "a+-b" ],
|
||||||
|
[ "a + - b--", "a+-b--" ],
|
||||||
|
[ "a + - b++", "a+-b++" ],
|
||||||
|
[ "a - ++b", "a-++b" ],
|
||||||
|
[ "a - --b", "a- --b" ],
|
||||||
|
[ "a - b", "a-b" ],
|
||||||
|
[ "a - b--", "a-b--" ],
|
||||||
|
[ "a - b++", "a-b++" ],
|
||||||
|
[ "a - + ++b", "a-+ ++b" ],
|
||||||
|
[ "a - + --b", "a-+--b" ],
|
||||||
|
[ "a - + b", "a-+b" ],
|
||||||
|
[ "a - + b--", "a-+b--" ],
|
||||||
|
[ "a - + b++", "a-+b++" ],
|
||||||
|
[ "a - - ++b", "a- -++b" ],
|
||||||
|
[ "a - - --b", "a- - --b" ],
|
||||||
|
[ "a - - b", "a- -b" ],
|
||||||
|
[ "a - - b--", "a- -b--" ],
|
||||||
|
[ "a - - b++", "a- -b++" ],
|
||||||
|
[ "a-- + ++b", "a--+ ++b" ],
|
||||||
|
[ "a-- + --b", "a--+--b" ],
|
||||||
|
[ "a-- + b", "a--+b" ],
|
||||||
|
[ "a-- + b--", "a--+b--" ],
|
||||||
|
[ "a-- + b++", "a--+b++" ],
|
||||||
|
[ "a-- + + ++b", "a--+ + ++b" ],
|
||||||
|
[ "a-- + + --b", "a--+ +--b" ],
|
||||||
|
[ "a-- + + b", "a--+ +b" ],
|
||||||
|
[ "a-- + + b--", "a--+ +b--" ],
|
||||||
|
[ "a-- + + b++", "a--+ +b++" ],
|
||||||
|
[ "a-- + - ++b", "a--+-++b" ],
|
||||||
|
[ "a-- + - --b", "a--+- --b" ],
|
||||||
|
[ "a-- + - b", "a--+-b" ],
|
||||||
|
[ "a-- + - b--", "a--+-b--" ],
|
||||||
|
[ "a-- + - b++", "a--+-b++" ],
|
||||||
|
[ "a-- - ++b", "a---++b" ],
|
||||||
|
[ "a-- - --b", "a--- --b" ],
|
||||||
|
[ "a-- - b", "a---b" ],
|
||||||
|
[ "a-- - b--", "a---b--" ],
|
||||||
|
[ "a-- - b++", "a---b++" ],
|
||||||
|
[ "a-- - + ++b", "a---+ ++b" ],
|
||||||
|
[ "a-- - + --b", "a---+--b" ],
|
||||||
|
[ "a-- - + b", "a---+b" ],
|
||||||
|
[ "a-- - + b--", "a---+b--" ],
|
||||||
|
[ "a-- - + b++", "a---+b++" ],
|
||||||
|
[ "a-- - - ++b", "a--- -++b" ],
|
||||||
|
[ "a-- - - --b", "a--- - --b" ],
|
||||||
|
[ "a-- - - b", "a--- -b" ],
|
||||||
|
[ "a-- - - b--", "a--- -b--" ],
|
||||||
|
[ "a-- - - b++", "a--- -b++" ],
|
||||||
|
[ "a++ + ++b", "a+++ ++b" ],
|
||||||
|
[ "a++ + --b", "a+++--b" ],
|
||||||
|
[ "a++ + b", "a+++b" ],
|
||||||
|
[ "a++ + b--", "a+++b--" ],
|
||||||
|
[ "a++ + b++", "a+++b++" ],
|
||||||
|
[ "a++ + + ++b", "a+++ + ++b" ],
|
||||||
|
[ "a++ + + --b", "a+++ +--b" ],
|
||||||
|
[ "a++ + + b", "a+++ +b" ],
|
||||||
|
[ "a++ + + b--", "a+++ +b--" ],
|
||||||
|
[ "a++ + + b++", "a+++ +b++" ],
|
||||||
|
[ "a++ + - ++b", "a+++-++b" ],
|
||||||
|
[ "a++ + - --b", "a+++- --b" ],
|
||||||
|
[ "a++ + - b", "a+++-b" ],
|
||||||
|
[ "a++ + - b--", "a+++-b--" ],
|
||||||
|
[ "a++ + - b++", "a+++-b++" ],
|
||||||
|
[ "a++ - ++b", "a++-++b" ],
|
||||||
|
[ "a++ - --b", "a++- --b" ],
|
||||||
|
[ "a++ - b", "a++-b" ],
|
||||||
|
[ "a++ - b--", "a++-b--" ],
|
||||||
|
[ "a++ - b++", "a++-b++" ],
|
||||||
|
[ "a++ - + ++b", "a++-+ ++b" ],
|
||||||
|
[ "a++ - + --b", "a++-+--b" ],
|
||||||
|
[ "a++ - + b", "a++-+b" ],
|
||||||
|
[ "a++ - + b--", "a++-+b--" ],
|
||||||
|
[ "a++ - + b++", "a++-+b++" ],
|
||||||
|
[ "a++ - - ++b", "a++- -++b" ],
|
||||||
|
[ "a++ - - --b", "a++- - --b" ],
|
||||||
|
[ "a++ - - b", "a++- -b" ],
|
||||||
|
[ "a++ - - b--", "a++- -b--" ],
|
||||||
|
[ "a++ - - b++", "a++- -b++" ],
|
||||||
|
[ "+ ++a + ++b", "+ ++a+ ++b" ],
|
||||||
|
[ "+ ++a + --b", "+ ++a+--b" ],
|
||||||
|
[ "+ ++a + b", "+ ++a+b" ],
|
||||||
|
[ "+ ++a + b--", "+ ++a+b--" ],
|
||||||
|
[ "+ ++a + b++", "+ ++a+b++" ],
|
||||||
|
[ "+ ++a + + ++b", "+ ++a+ + ++b" ],
|
||||||
|
[ "+ ++a + + --b", "+ ++a+ +--b" ],
|
||||||
|
[ "+ ++a + + b", "+ ++a+ +b" ],
|
||||||
|
[ "+ ++a + + b--", "+ ++a+ +b--" ],
|
||||||
|
[ "+ ++a + + b++", "+ ++a+ +b++" ],
|
||||||
|
[ "+ ++a + - ++b", "+ ++a+-++b" ],
|
||||||
|
[ "+ ++a + - --b", "+ ++a+- --b" ],
|
||||||
|
[ "+ ++a + - b", "+ ++a+-b" ],
|
||||||
|
[ "+ ++a + - b--", "+ ++a+-b--" ],
|
||||||
|
[ "+ ++a + - b++", "+ ++a+-b++" ],
|
||||||
|
[ "+ ++a - ++b", "+ ++a-++b" ],
|
||||||
|
[ "+ ++a - --b", "+ ++a- --b" ],
|
||||||
|
[ "+ ++a - b", "+ ++a-b" ],
|
||||||
|
[ "+ ++a - b--", "+ ++a-b--" ],
|
||||||
|
[ "+ ++a - b++", "+ ++a-b++" ],
|
||||||
|
[ "+ ++a - + ++b", "+ ++a-+ ++b" ],
|
||||||
|
[ "+ ++a - + --b", "+ ++a-+--b" ],
|
||||||
|
[ "+ ++a - + b", "+ ++a-+b" ],
|
||||||
|
[ "+ ++a - + b--", "+ ++a-+b--" ],
|
||||||
|
[ "+ ++a - + b++", "+ ++a-+b++" ],
|
||||||
|
[ "+ ++a - - ++b", "+ ++a- -++b" ],
|
||||||
|
[ "+ ++a - - --b", "+ ++a- - --b" ],
|
||||||
|
[ "+ ++a - - b", "+ ++a- -b" ],
|
||||||
|
[ "+ ++a - - b--", "+ ++a- -b--" ],
|
||||||
|
[ "+ ++a - - b++", "+ ++a- -b++" ],
|
||||||
|
[ "+ --a + ++b", "+--a+ ++b" ],
|
||||||
|
[ "+ --a + --b", "+--a+--b" ],
|
||||||
|
[ "+ --a + b", "+--a+b" ],
|
||||||
|
[ "+ --a + b--", "+--a+b--" ],
|
||||||
|
[ "+ --a + b++", "+--a+b++" ],
|
||||||
|
[ "+ --a + + ++b", "+--a+ + ++b" ],
|
||||||
|
[ "+ --a + + --b", "+--a+ +--b" ],
|
||||||
|
[ "+ --a + + b", "+--a+ +b" ],
|
||||||
|
[ "+ --a + + b--", "+--a+ +b--" ],
|
||||||
|
[ "+ --a + + b++", "+--a+ +b++" ],
|
||||||
|
[ "+ --a + - ++b", "+--a+-++b" ],
|
||||||
|
[ "+ --a + - --b", "+--a+- --b" ],
|
||||||
|
[ "+ --a + - b", "+--a+-b" ],
|
||||||
|
[ "+ --a + - b--", "+--a+-b--" ],
|
||||||
|
[ "+ --a + - b++", "+--a+-b++" ],
|
||||||
|
[ "+ --a - ++b", "+--a-++b" ],
|
||||||
|
[ "+ --a - --b", "+--a- --b" ],
|
||||||
|
[ "+ --a - b", "+--a-b" ],
|
||||||
|
[ "+ --a - b--", "+--a-b--" ],
|
||||||
|
[ "+ --a - b++", "+--a-b++" ],
|
||||||
|
[ "+ --a - + ++b", "+--a-+ ++b" ],
|
||||||
|
[ "+ --a - + --b", "+--a-+--b" ],
|
||||||
|
[ "+ --a - + b", "+--a-+b" ],
|
||||||
|
[ "+ --a - + b--", "+--a-+b--" ],
|
||||||
|
[ "+ --a - + b++", "+--a-+b++" ],
|
||||||
|
[ "+ --a - - ++b", "+--a- -++b" ],
|
||||||
|
[ "+ --a - - --b", "+--a- - --b" ],
|
||||||
|
[ "+ --a - - b", "+--a- -b" ],
|
||||||
|
[ "+ --a - - b--", "+--a- -b--" ],
|
||||||
|
[ "+ --a - - b++", "+--a- -b++" ],
|
||||||
|
[ "+ a + ++b", "+a+ ++b" ],
|
||||||
|
[ "+ a + --b", "+a+--b" ],
|
||||||
|
[ "+ a + b", "+a+b" ],
|
||||||
|
[ "+ a + b--", "+a+b--" ],
|
||||||
|
[ "+ a + b++", "+a+b++" ],
|
||||||
|
[ "+ a + + ++b", "+a+ + ++b" ],
|
||||||
|
[ "+ a + + --b", "+a+ +--b" ],
|
||||||
|
[ "+ a + + b", "+a+ +b" ],
|
||||||
|
[ "+ a + + b--", "+a+ +b--" ],
|
||||||
|
[ "+ a + + b++", "+a+ +b++" ],
|
||||||
|
[ "+ a + - ++b", "+a+-++b" ],
|
||||||
|
[ "+ a + - --b", "+a+- --b" ],
|
||||||
|
[ "+ a + - b", "+a+-b" ],
|
||||||
|
[ "+ a + - b--", "+a+-b--" ],
|
||||||
|
[ "+ a + - b++", "+a+-b++" ],
|
||||||
|
[ "+ a - ++b", "+a-++b" ],
|
||||||
|
[ "+ a - --b", "+a- --b" ],
|
||||||
|
[ "+ a - b", "+a-b" ],
|
||||||
|
[ "+ a - b--", "+a-b--" ],
|
||||||
|
[ "+ a - b++", "+a-b++" ],
|
||||||
|
[ "+ a - + ++b", "+a-+ ++b" ],
|
||||||
|
[ "+ a - + --b", "+a-+--b" ],
|
||||||
|
[ "+ a - + b", "+a-+b" ],
|
||||||
|
[ "+ a - + b--", "+a-+b--" ],
|
||||||
|
[ "+ a - + b++", "+a-+b++" ],
|
||||||
|
[ "+ a - - ++b", "+a- -++b" ],
|
||||||
|
[ "+ a - - --b", "+a- - --b" ],
|
||||||
|
[ "+ a - - b", "+a- -b" ],
|
||||||
|
[ "+ a - - b--", "+a- -b--" ],
|
||||||
|
[ "+ a - - b++", "+a- -b++" ],
|
||||||
|
[ "+ a-- + ++b", "+a--+ ++b" ],
|
||||||
|
[ "+ a-- + --b", "+a--+--b" ],
|
||||||
|
[ "+ a-- + b", "+a--+b" ],
|
||||||
|
[ "+ a-- + b--", "+a--+b--" ],
|
||||||
|
[ "+ a-- + b++", "+a--+b++" ],
|
||||||
|
[ "+ a-- + + ++b", "+a--+ + ++b" ],
|
||||||
|
[ "+ a-- + + --b", "+a--+ +--b" ],
|
||||||
|
[ "+ a-- + + b", "+a--+ +b" ],
|
||||||
|
[ "+ a-- + + b--", "+a--+ +b--" ],
|
||||||
|
[ "+ a-- + + b++", "+a--+ +b++" ],
|
||||||
|
[ "+ a-- + - ++b", "+a--+-++b" ],
|
||||||
|
[ "+ a-- + - --b", "+a--+- --b" ],
|
||||||
|
[ "+ a-- + - b", "+a--+-b" ],
|
||||||
|
[ "+ a-- + - b--", "+a--+-b--" ],
|
||||||
|
[ "+ a-- + - b++", "+a--+-b++" ],
|
||||||
|
[ "+ a-- - ++b", "+a---++b" ],
|
||||||
|
[ "+ a-- - --b", "+a--- --b" ],
|
||||||
|
[ "+ a-- - b", "+a---b" ],
|
||||||
|
[ "+ a-- - b--", "+a---b--" ],
|
||||||
|
[ "+ a-- - b++", "+a---b++" ],
|
||||||
|
[ "+ a-- - + ++b", "+a---+ ++b" ],
|
||||||
|
[ "+ a-- - + --b", "+a---+--b" ],
|
||||||
|
[ "+ a-- - + b", "+a---+b" ],
|
||||||
|
[ "+ a-- - + b--", "+a---+b--" ],
|
||||||
|
[ "+ a-- - + b++", "+a---+b++" ],
|
||||||
|
[ "+ a-- - - ++b", "+a--- -++b" ],
|
||||||
|
[ "+ a-- - - --b", "+a--- - --b" ],
|
||||||
|
[ "+ a-- - - b", "+a--- -b" ],
|
||||||
|
[ "+ a-- - - b--", "+a--- -b--" ],
|
||||||
|
[ "+ a-- - - b++", "+a--- -b++" ],
|
||||||
|
[ "+ a++ + ++b", "+a+++ ++b" ],
|
||||||
|
[ "+ a++ + --b", "+a+++--b" ],
|
||||||
|
[ "+ a++ + b", "+a+++b" ],
|
||||||
|
[ "+ a++ + b--", "+a+++b--" ],
|
||||||
|
[ "+ a++ + b++", "+a+++b++" ],
|
||||||
|
[ "+ a++ + + ++b", "+a+++ + ++b" ],
|
||||||
|
[ "+ a++ + + --b", "+a+++ +--b" ],
|
||||||
|
[ "+ a++ + + b", "+a+++ +b" ],
|
||||||
|
[ "+ a++ + + b--", "+a+++ +b--" ],
|
||||||
|
[ "+ a++ + + b++", "+a+++ +b++" ],
|
||||||
|
[ "+ a++ + - ++b", "+a+++-++b" ],
|
||||||
|
[ "+ a++ + - --b", "+a+++- --b" ],
|
||||||
|
[ "+ a++ + - b", "+a+++-b" ],
|
||||||
|
[ "+ a++ + - b--", "+a+++-b--" ],
|
||||||
|
[ "+ a++ + - b++", "+a+++-b++" ],
|
||||||
|
[ "+ a++ - ++b", "+a++-++b" ],
|
||||||
|
[ "+ a++ - --b", "+a++- --b" ],
|
||||||
|
[ "+ a++ - b", "+a++-b" ],
|
||||||
|
[ "+ a++ - b--", "+a++-b--" ],
|
||||||
|
[ "+ a++ - b++", "+a++-b++" ],
|
||||||
|
[ "+ a++ - + ++b", "+a++-+ ++b" ],
|
||||||
|
[ "+ a++ - + --b", "+a++-+--b" ],
|
||||||
|
[ "+ a++ - + b", "+a++-+b" ],
|
||||||
|
[ "+ a++ - + b--", "+a++-+b--" ],
|
||||||
|
[ "+ a++ - + b++", "+a++-+b++" ],
|
||||||
|
[ "+ a++ - - ++b", "+a++- -++b" ],
|
||||||
|
[ "+ a++ - - --b", "+a++- - --b" ],
|
||||||
|
[ "+ a++ - - b", "+a++- -b" ],
|
||||||
|
[ "+ a++ - - b--", "+a++- -b--" ],
|
||||||
|
[ "+ a++ - - b++", "+a++- -b++" ],
|
||||||
|
[ "- ++a + ++b", "-++a+ ++b" ],
|
||||||
|
[ "- ++a + --b", "-++a+--b" ],
|
||||||
|
[ "- ++a + b", "-++a+b" ],
|
||||||
|
[ "- ++a + b--", "-++a+b--" ],
|
||||||
|
[ "- ++a + b++", "-++a+b++" ],
|
||||||
|
[ "- ++a + + ++b", "-++a+ + ++b" ],
|
||||||
|
[ "- ++a + + --b", "-++a+ +--b" ],
|
||||||
|
[ "- ++a + + b", "-++a+ +b" ],
|
||||||
|
[ "- ++a + + b--", "-++a+ +b--" ],
|
||||||
|
[ "- ++a + + b++", "-++a+ +b++" ],
|
||||||
|
[ "- ++a + - ++b", "-++a+-++b" ],
|
||||||
|
[ "- ++a + - --b", "-++a+- --b" ],
|
||||||
|
[ "- ++a + - b", "-++a+-b" ],
|
||||||
|
[ "- ++a + - b--", "-++a+-b--" ],
|
||||||
|
[ "- ++a + - b++", "-++a+-b++" ],
|
||||||
|
[ "- ++a - ++b", "-++a-++b" ],
|
||||||
|
[ "- ++a - --b", "-++a- --b" ],
|
||||||
|
[ "- ++a - b", "-++a-b" ],
|
||||||
|
[ "- ++a - b--", "-++a-b--" ],
|
||||||
|
[ "- ++a - b++", "-++a-b++" ],
|
||||||
|
[ "- ++a - + ++b", "-++a-+ ++b" ],
|
||||||
|
[ "- ++a - + --b", "-++a-+--b" ],
|
||||||
|
[ "- ++a - + b", "-++a-+b" ],
|
||||||
|
[ "- ++a - + b--", "-++a-+b--" ],
|
||||||
|
[ "- ++a - + b++", "-++a-+b++" ],
|
||||||
|
[ "- ++a - - ++b", "-++a- -++b" ],
|
||||||
|
[ "- ++a - - --b", "-++a- - --b" ],
|
||||||
|
[ "- ++a - - b", "-++a- -b" ],
|
||||||
|
[ "- ++a - - b--", "-++a- -b--" ],
|
||||||
|
[ "- ++a - - b++", "-++a- -b++" ],
|
||||||
|
[ "- --a + ++b", "- --a+ ++b" ],
|
||||||
|
[ "- --a + --b", "- --a+--b" ],
|
||||||
|
[ "- --a + b", "- --a+b" ],
|
||||||
|
[ "- --a + b--", "- --a+b--" ],
|
||||||
|
[ "- --a + b++", "- --a+b++" ],
|
||||||
|
[ "- --a + + ++b", "- --a+ + ++b" ],
|
||||||
|
[ "- --a + + --b", "- --a+ +--b" ],
|
||||||
|
[ "- --a + + b", "- --a+ +b" ],
|
||||||
|
[ "- --a + + b--", "- --a+ +b--" ],
|
||||||
|
[ "- --a + + b++", "- --a+ +b++" ],
|
||||||
|
[ "- --a + - ++b", "- --a+-++b" ],
|
||||||
|
[ "- --a + - --b", "- --a+- --b" ],
|
||||||
|
[ "- --a + - b", "- --a+-b" ],
|
||||||
|
[ "- --a + - b--", "- --a+-b--" ],
|
||||||
|
[ "- --a + - b++", "- --a+-b++" ],
|
||||||
|
[ "- --a - ++b", "- --a-++b" ],
|
||||||
|
[ "- --a - --b", "- --a- --b" ],
|
||||||
|
[ "- --a - b", "- --a-b" ],
|
||||||
|
[ "- --a - b--", "- --a-b--" ],
|
||||||
|
[ "- --a - b++", "- --a-b++" ],
|
||||||
|
[ "- --a - + ++b", "- --a-+ ++b" ],
|
||||||
|
[ "- --a - + --b", "- --a-+--b" ],
|
||||||
|
[ "- --a - + b", "- --a-+b" ],
|
||||||
|
[ "- --a - + b--", "- --a-+b--" ],
|
||||||
|
[ "- --a - + b++", "- --a-+b++" ],
|
||||||
|
[ "- --a - - ++b", "- --a- -++b" ],
|
||||||
|
[ "- --a - - --b", "- --a- - --b" ],
|
||||||
|
[ "- --a - - b", "- --a- -b" ],
|
||||||
|
[ "- --a - - b--", "- --a- -b--" ],
|
||||||
|
[ "- --a - - b++", "- --a- -b++" ],
|
||||||
|
[ "- a + ++b", "-a+ ++b" ],
|
||||||
|
[ "- a + --b", "-a+--b" ],
|
||||||
|
[ "- a + b", "-a+b" ],
|
||||||
|
[ "- a + b--", "-a+b--" ],
|
||||||
|
[ "- a + b++", "-a+b++" ],
|
||||||
|
[ "- a + + ++b", "-a+ + ++b" ],
|
||||||
|
[ "- a + + --b", "-a+ +--b" ],
|
||||||
|
[ "- a + + b", "-a+ +b" ],
|
||||||
|
[ "- a + + b--", "-a+ +b--" ],
|
||||||
|
[ "- a + + b++", "-a+ +b++" ],
|
||||||
|
[ "- a + - ++b", "-a+-++b" ],
|
||||||
|
[ "- a + - --b", "-a+- --b" ],
|
||||||
|
[ "- a + - b", "-a+-b" ],
|
||||||
|
[ "- a + - b--", "-a+-b--" ],
|
||||||
|
[ "- a + - b++", "-a+-b++" ],
|
||||||
|
[ "- a - ++b", "-a-++b" ],
|
||||||
|
[ "- a - --b", "-a- --b" ],
|
||||||
|
[ "- a - b", "-a-b" ],
|
||||||
|
[ "- a - b--", "-a-b--" ],
|
||||||
|
[ "- a - b++", "-a-b++" ],
|
||||||
|
[ "- a - + ++b", "-a-+ ++b" ],
|
||||||
|
[ "- a - + --b", "-a-+--b" ],
|
||||||
|
[ "- a - + b", "-a-+b" ],
|
||||||
|
[ "- a - + b--", "-a-+b--" ],
|
||||||
|
[ "- a - + b++", "-a-+b++" ],
|
||||||
|
[ "- a - - ++b", "-a- -++b" ],
|
||||||
|
[ "- a - - --b", "-a- - --b" ],
|
||||||
|
[ "- a - - b", "-a- -b" ],
|
||||||
|
[ "- a - - b--", "-a- -b--" ],
|
||||||
|
[ "- a - - b++", "-a- -b++" ],
|
||||||
|
[ "- a-- + ++b", "-a--+ ++b" ],
|
||||||
|
[ "- a-- + --b", "-a--+--b" ],
|
||||||
|
[ "- a-- + b", "-a--+b" ],
|
||||||
|
[ "- a-- + b--", "-a--+b--" ],
|
||||||
|
[ "- a-- + b++", "-a--+b++" ],
|
||||||
|
[ "- a-- + + ++b", "-a--+ + ++b" ],
|
||||||
|
[ "- a-- + + --b", "-a--+ +--b" ],
|
||||||
|
[ "- a-- + + b", "-a--+ +b" ],
|
||||||
|
[ "- a-- + + b--", "-a--+ +b--" ],
|
||||||
|
[ "- a-- + + b++", "-a--+ +b++" ],
|
||||||
|
[ "- a-- + - ++b", "-a--+-++b" ],
|
||||||
|
[ "- a-- + - --b", "-a--+- --b" ],
|
||||||
|
[ "- a-- + - b", "-a--+-b" ],
|
||||||
|
[ "- a-- + - b--", "-a--+-b--" ],
|
||||||
|
[ "- a-- + - b++", "-a--+-b++" ],
|
||||||
|
[ "- a-- - ++b", "-a---++b" ],
|
||||||
|
[ "- a-- - --b", "-a--- --b" ],
|
||||||
|
[ "- a-- - b", "-a---b" ],
|
||||||
|
[ "- a-- - b--", "-a---b--" ],
|
||||||
|
[ "- a-- - b++", "-a---b++" ],
|
||||||
|
[ "- a-- - + ++b", "-a---+ ++b" ],
|
||||||
|
[ "- a-- - + --b", "-a---+--b" ],
|
||||||
|
[ "- a-- - + b", "-a---+b" ],
|
||||||
|
[ "- a-- - + b--", "-a---+b--" ],
|
||||||
|
[ "- a-- - + b++", "-a---+b++" ],
|
||||||
|
[ "- a-- - - ++b", "-a--- -++b" ],
|
||||||
|
[ "- a-- - - --b", "-a--- - --b" ],
|
||||||
|
[ "- a-- - - b", "-a--- -b" ],
|
||||||
|
[ "- a-- - - b--", "-a--- -b--" ],
|
||||||
|
[ "- a-- - - b++", "-a--- -b++" ],
|
||||||
|
[ "- a++ + ++b", "-a+++ ++b" ],
|
||||||
|
[ "- a++ + --b", "-a+++--b" ],
|
||||||
|
[ "- a++ + b", "-a+++b" ],
|
||||||
|
[ "- a++ + b--", "-a+++b--" ],
|
||||||
|
[ "- a++ + b++", "-a+++b++" ],
|
||||||
|
[ "- a++ + + ++b", "-a+++ + ++b" ],
|
||||||
|
[ "- a++ + + --b", "-a+++ +--b" ],
|
||||||
|
[ "- a++ + + b", "-a+++ +b" ],
|
||||||
|
[ "- a++ + + b--", "-a+++ +b--" ],
|
||||||
|
[ "- a++ + + b++", "-a+++ +b++" ],
|
||||||
|
[ "- a++ + - ++b", "-a+++-++b" ],
|
||||||
|
[ "- a++ + - --b", "-a+++- --b" ],
|
||||||
|
[ "- a++ + - b", "-a+++-b" ],
|
||||||
|
[ "- a++ + - b--", "-a+++-b--" ],
|
||||||
|
[ "- a++ + - b++", "-a+++-b++" ],
|
||||||
|
[ "- a++ - ++b", "-a++-++b" ],
|
||||||
|
[ "- a++ - --b", "-a++- --b" ],
|
||||||
|
[ "- a++ - b", "-a++-b" ],
|
||||||
|
[ "- a++ - b--", "-a++-b--" ],
|
||||||
|
[ "- a++ - b++", "-a++-b++" ],
|
||||||
|
[ "- a++ - + ++b", "-a++-+ ++b" ],
|
||||||
|
[ "- a++ - + --b", "-a++-+--b" ],
|
||||||
|
[ "- a++ - + b", "-a++-+b" ],
|
||||||
|
[ "- a++ - + b--", "-a++-+b--" ],
|
||||||
|
[ "- a++ - + b++", "-a++-+b++" ],
|
||||||
|
[ "- a++ - - ++b", "-a++- -++b" ],
|
||||||
|
[ "- a++ - - --b", "-a++- - --b" ],
|
||||||
|
[ "- a++ - - b", "-a++- -b" ],
|
||||||
|
[ "- a++ - - b--", "-a++- -b--" ],
|
||||||
|
[ "- a++ - - b++", "-a++- -b++" ],
|
||||||
|
].forEach(function(exp) {
|
||||||
|
assert.strictEqual(UglifyJS.parse(exp[0]).print_to_string(), exp[1] + ";");
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
23
test/mocha/screw-ie8.js
Normal file
23
test/mocha/screw-ie8.js
Normal file
@@ -0,0 +1,23 @@
|
|||||||
|
var assert = require("assert");
|
||||||
|
var uglify = require("../../");
|
||||||
|
|
||||||
|
describe("screw-ie8", function () {
|
||||||
|
it("Should be able to minify() with undefined as catch parameter in a try...catch statement", function () {
|
||||||
|
assert.strictEqual(
|
||||||
|
uglify.minify(
|
||||||
|
"function a(b){\
|
||||||
|
try {\
|
||||||
|
throw 'Stuff';\
|
||||||
|
} catch (undefined) {\
|
||||||
|
console.log('caught: ' + undefined);\
|
||||||
|
}\
|
||||||
|
console.log('undefined is ' + undefined);\
|
||||||
|
return b === undefined;\
|
||||||
|
};", {
|
||||||
|
fromString: true
|
||||||
|
}
|
||||||
|
).code,
|
||||||
|
'function a(o){try{throw"Stuff"}catch(o){console.log("caught: "+o)}return console.log("undefined is "+void 0),void 0===o}'
|
||||||
|
);
|
||||||
|
});
|
||||||
|
});
|
||||||
124
test/mocha/spidermonkey.js
Normal file
124
test/mocha/spidermonkey.js
Normal file
@@ -0,0 +1,124 @@
|
|||||||
|
var assert = require("assert");
|
||||||
|
var exec = require("child_process").exec;
|
||||||
|
var uglify = require("../../");
|
||||||
|
|
||||||
|
describe("spidermonkey export/import sanity test", function() {
|
||||||
|
it("should produce a functional build when using --self with spidermonkey", function (done) {
|
||||||
|
this.timeout(20000);
|
||||||
|
|
||||||
|
var uglifyjs = '"' + process.argv[0] + '" bin/uglifyjs';
|
||||||
|
var command = uglifyjs + " --self -cm --wrap SpiderUglify --dump-spidermonkey-ast | " +
|
||||||
|
uglifyjs + " --spidermonkey -cm";
|
||||||
|
|
||||||
|
exec(command, function (err, stdout) {
|
||||||
|
if (err) throw err;
|
||||||
|
|
||||||
|
eval(stdout);
|
||||||
|
assert.strictEqual(typeof SpiderUglify, "object");
|
||||||
|
|
||||||
|
var ast = SpiderUglify.parse("foo([true,,2+3]);");
|
||||||
|
assert.strictEqual(true, ast instanceof SpiderUglify.AST_Node);
|
||||||
|
|
||||||
|
ast.figure_out_scope();
|
||||||
|
ast = SpiderUglify.Compressor({}).compress(ast);
|
||||||
|
assert.strictEqual(true, ast instanceof SpiderUglify.AST_Node);
|
||||||
|
|
||||||
|
var stream = SpiderUglify.OutputStream({});
|
||||||
|
ast.print(stream);
|
||||||
|
var code = stream.toString();
|
||||||
|
assert.strictEqual(code, "foo([!0,,5]);");
|
||||||
|
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it("Should judge between directives and strings correctly on import", function() {
|
||||||
|
var tests = [
|
||||||
|
{
|
||||||
|
input: '"use strict";;"use sloppy"',
|
||||||
|
directives: 1,
|
||||||
|
strings: 1
|
||||||
|
},
|
||||||
|
{
|
||||||
|
input: ';"use strict"',
|
||||||
|
directives: 0,
|
||||||
|
strings: 1
|
||||||
|
},
|
||||||
|
{
|
||||||
|
input: '"use strict"; "use something else";',
|
||||||
|
directives: 2,
|
||||||
|
strings: 0
|
||||||
|
},
|
||||||
|
{
|
||||||
|
input: 'function foo() {"use strict";;"use sloppy" }',
|
||||||
|
directives: 1,
|
||||||
|
strings: 1
|
||||||
|
},
|
||||||
|
{
|
||||||
|
input: 'function foo() {;"use strict" }',
|
||||||
|
directives: 0,
|
||||||
|
strings: 1
|
||||||
|
},
|
||||||
|
{
|
||||||
|
input: 'function foo() {"use strict"; "use something else"; }',
|
||||||
|
directives: 2,
|
||||||
|
strings: 0
|
||||||
|
},
|
||||||
|
{
|
||||||
|
input: 'var foo = function() {"use strict";;"use sloppy" }',
|
||||||
|
directives: 1,
|
||||||
|
strings: 1
|
||||||
|
},
|
||||||
|
{
|
||||||
|
input: 'var foo = function() {;"use strict" }',
|
||||||
|
directives: 0,
|
||||||
|
strings: 1
|
||||||
|
},
|
||||||
|
{
|
||||||
|
input: 'var foo = function() {"use strict"; "use something else"; }',
|
||||||
|
directives: 2,
|
||||||
|
strings: 0
|
||||||
|
},
|
||||||
|
{
|
||||||
|
input: '{"use strict";;"use sloppy" }',
|
||||||
|
directives: 0,
|
||||||
|
strings: 2
|
||||||
|
},
|
||||||
|
{
|
||||||
|
input: '{;"use strict" }',
|
||||||
|
directives: 0,
|
||||||
|
strings: 1
|
||||||
|
},
|
||||||
|
{
|
||||||
|
input: '{"use strict"; "use something else"; }',
|
||||||
|
directives: 0,
|
||||||
|
strings: 2
|
||||||
|
}
|
||||||
|
];
|
||||||
|
|
||||||
|
var counter_directives;
|
||||||
|
var counter_strings;
|
||||||
|
|
||||||
|
var checkWalker = new uglify.TreeWalker(function(node, descend) {
|
||||||
|
if (node instanceof uglify.AST_String) {
|
||||||
|
counter_strings++;
|
||||||
|
} else if (node instanceof uglify.AST_Directive) {
|
||||||
|
counter_directives++;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
for (var i = 0; i < tests.length; i++) {
|
||||||
|
counter_directives = 0;
|
||||||
|
counter_strings = 0;
|
||||||
|
|
||||||
|
var ast = uglify.parse(tests[i].input);
|
||||||
|
var moz_ast = ast.to_mozilla_ast();
|
||||||
|
var from_moz_ast = uglify.AST_Node.from_mozilla_ast(moz_ast);
|
||||||
|
|
||||||
|
from_moz_ast.walk(checkWalker);
|
||||||
|
|
||||||
|
assert.strictEqual(counter_directives, tests[i].directives, "Directives count mismatch for test " + tests[i].input);
|
||||||
|
assert.strictEqual(counter_strings, tests[i].strings, "String count mismatch for test " + tests[i].input);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
@@ -49,7 +49,7 @@ describe("String literals", function() {
|
|||||||
|
|
||||||
var error = function(e) {
|
var error = function(e) {
|
||||||
return e instanceof UglifyJS.JS_Parse_Error &&
|
return e instanceof UglifyJS.JS_Parse_Error &&
|
||||||
e.message === "Octal literals are not allowed in strict mode";
|
e.message === "Legacy octal escape sequences are not allowed in strict mode";
|
||||||
}
|
}
|
||||||
|
|
||||||
for (var input in inputs) {
|
for (var input in inputs) {
|
||||||
|
|||||||
@@ -2,15 +2,22 @@ var assert = require("assert");
|
|||||||
var uglify = require("../../");
|
var uglify = require("../../");
|
||||||
|
|
||||||
describe("With", function() {
|
describe("With", function() {
|
||||||
it ("Should throw syntaxError when using with statement in strict mode", function() {
|
it("Should throw syntaxError when using with statement in strict mode", function() {
|
||||||
var code = '"use strict";\nthrow NotEarlyError;\nwith ({}) { }';
|
var code = '"use strict";\nthrow NotEarlyError;\nwith ({}) { }';
|
||||||
var test = function() {
|
var test = function() {
|
||||||
uglify.parse(code);
|
uglify.parse(code);
|
||||||
}
|
}
|
||||||
var error = function(e) {
|
var error = function(e) {
|
||||||
return e instanceof uglify.JS_Parse_Error &&
|
return e instanceof uglify.JS_Parse_Error &&
|
||||||
e.message === "SyntaxError: Strict mode may not include a with statement";
|
e.message === "Strict mode may not include a with statement";
|
||||||
}
|
}
|
||||||
assert.throws(test, error);
|
assert.throws(test, error);
|
||||||
});
|
});
|
||||||
});
|
it("Should set uses_with for scopes involving With statements", function() {
|
||||||
|
var ast = uglify.parse("with(e) {f(1, 2)}");
|
||||||
|
ast.figure_out_scope();
|
||||||
|
assert.equal(ast.uses_with, true);
|
||||||
|
assert.equal(ast.body[0].expression.scope.uses_with, true);
|
||||||
|
assert.equal(ast.body[0].body.body[0].body.expression.scope.uses_with, true);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|||||||
@@ -70,12 +70,12 @@ function test_directory(dir) {
|
|||||||
return path.resolve(tests_dir, dir);
|
return path.resolve(tests_dir, dir);
|
||||||
}
|
}
|
||||||
|
|
||||||
function as_toplevel(input) {
|
function as_toplevel(input, mangle_options) {
|
||||||
if (input instanceof U.AST_BlockStatement) input = input.body;
|
if (input instanceof U.AST_BlockStatement) input = input.body;
|
||||||
else if (input instanceof U.AST_Statement) input = [ input ];
|
else if (input instanceof U.AST_Statement) input = [ input ];
|
||||||
else throw new Error("Unsupported input syntax");
|
else throw new Error("Unsupported input syntax");
|
||||||
var toplevel = new U.AST_Toplevel({ body: input });
|
var toplevel = new U.AST_Toplevel({ body: input });
|
||||||
toplevel.figure_out_scope();
|
toplevel.figure_out_scope(mangle_options);
|
||||||
return toplevel;
|
return toplevel;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -103,11 +103,11 @@ function run_compress_tests() {
|
|||||||
var output_options = test.beautify || {};
|
var output_options = test.beautify || {};
|
||||||
var expect;
|
var expect;
|
||||||
if (test.expect) {
|
if (test.expect) {
|
||||||
expect = make_code(as_toplevel(test.expect), output_options);
|
expect = make_code(as_toplevel(test.expect, test.mangle), output_options);
|
||||||
} else {
|
} else {
|
||||||
expect = test.expect_exact;
|
expect = test.expect_exact;
|
||||||
}
|
}
|
||||||
var input = as_toplevel(test.input);
|
var input = as_toplevel(test.input, test.mangle);
|
||||||
var input_code = make_code(test.input, {
|
var input_code = make_code(test.input, {
|
||||||
beautify: true,
|
beautify: true,
|
||||||
quote_style: 3,
|
quote_style: 3,
|
||||||
@@ -117,7 +117,7 @@ function run_compress_tests() {
|
|||||||
input = U.mangle_properties(input, test.mangle_props);
|
input = U.mangle_properties(input, test.mangle_props);
|
||||||
}
|
}
|
||||||
var output = cmp.compress(input);
|
var output = cmp.compress(input);
|
||||||
output.figure_out_scope();
|
output.figure_out_scope(test.mangle);
|
||||||
if (test.mangle) {
|
if (test.mangle) {
|
||||||
output.compute_char_frequency(test.mangle);
|
output.compute_char_frequency(test.mangle);
|
||||||
output.mangle_names(test.mangle);
|
output.mangle_names(test.mangle);
|
||||||
@@ -132,25 +132,39 @@ function run_compress_tests() {
|
|||||||
failures++;
|
failures++;
|
||||||
failed_files[file] = 1;
|
failed_files[file] = 1;
|
||||||
}
|
}
|
||||||
else if (test.expect_warnings) {
|
else {
|
||||||
U.AST_Node.warn_function = original_warn_function;
|
// expect == output
|
||||||
var expected_warnings = make_code(test.expect_warnings, {
|
try {
|
||||||
beautify: false,
|
var reparsed_ast = U.parse(output);
|
||||||
quote_style: 2, // force double quote to match JSON
|
} catch (ex) {
|
||||||
});
|
log("!!! Test matched expected result but cannot parse output\n---INPUT---\n{input}\n---OUTPUT---\n{output}\n--REPARSE ERROR--\n{error}\n\n", {
|
||||||
warnings_emitted = warnings_emitted.map(function(input) {
|
|
||||||
return input.split(process.cwd() + path.sep).join("").split(path.sep).join("/");
|
|
||||||
});
|
|
||||||
var actual_warnings = JSON.stringify(warnings_emitted);
|
|
||||||
if (expected_warnings != actual_warnings) {
|
|
||||||
log("!!! failed\n---INPUT---\n{input}\n---EXPECTED WARNINGS---\n{expected_warnings}\n---ACTUAL WARNINGS---\n{actual_warnings}\n\n", {
|
|
||||||
input: input_code,
|
input: input_code,
|
||||||
expected_warnings: expected_warnings,
|
output: output,
|
||||||
actual_warnings: actual_warnings,
|
error: ex.toString(),
|
||||||
});
|
});
|
||||||
failures++;
|
failures++;
|
||||||
failed_files[file] = 1;
|
failed_files[file] = 1;
|
||||||
}
|
}
|
||||||
|
if (test.expect_warnings) {
|
||||||
|
U.AST_Node.warn_function = original_warn_function;
|
||||||
|
var expected_warnings = make_code(test.expect_warnings, {
|
||||||
|
beautify: false,
|
||||||
|
quote_style: 2, // force double quote to match JSON
|
||||||
|
});
|
||||||
|
warnings_emitted = warnings_emitted.map(function(input) {
|
||||||
|
return input.split(process.cwd() + path.sep).join("").split(path.sep).join("/");
|
||||||
|
});
|
||||||
|
var actual_warnings = JSON.stringify(warnings_emitted);
|
||||||
|
if (expected_warnings != actual_warnings) {
|
||||||
|
log("!!! failed\n---INPUT---\n{input}\n---EXPECTED WARNINGS---\n{expected_warnings}\n---ACTUAL WARNINGS---\n{actual_warnings}\n\n", {
|
||||||
|
input: input_code,
|
||||||
|
expected_warnings: expected_warnings,
|
||||||
|
actual_warnings: actual_warnings,
|
||||||
|
});
|
||||||
|
failures++;
|
||||||
|
failed_files[file] = 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
var tests = parse_test(path.resolve(dir, file));
|
var tests = parse_test(path.resolve(dir, file));
|
||||||
@@ -180,6 +194,9 @@ function parse_test(file) {
|
|||||||
if (node instanceof U.AST_LabeledStatement
|
if (node instanceof U.AST_LabeledStatement
|
||||||
&& tw.parent() instanceof U.AST_Toplevel) {
|
&& tw.parent() instanceof U.AST_Toplevel) {
|
||||||
var name = node.label.name;
|
var name = node.label.name;
|
||||||
|
if (name in tests) {
|
||||||
|
throw new Error('Duplicated test name "' + name + '" in ' + file);
|
||||||
|
}
|
||||||
tests[name] = get_one_test(name, node.body);
|
tests[name] = get_one_test(name, node.body);
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|||||||
113
tools/node.js
113
tools/node.js
@@ -37,12 +37,24 @@ UglifyJS.AST_Node.warn_function = function(txt) {
|
|||||||
console.error("WARN: %s", txt);
|
console.error("WARN: %s", txt);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
function read_source_map(code) {
|
||||||
|
var match = /\n\/\/# sourceMappingURL=data:application\/json(;.*?)?;base64,(.*)/.exec(code);
|
||||||
|
if (!match) {
|
||||||
|
UglifyJS.AST_Node.warn("inline source map not found");
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
return JSON.parse(new Buffer(match[2], "base64"));
|
||||||
|
}
|
||||||
|
|
||||||
exports.minify = function(files, options) {
|
exports.minify = function(files, options) {
|
||||||
options = UglifyJS.defaults(options, {
|
options = UglifyJS.defaults(options, {
|
||||||
spidermonkey : false,
|
spidermonkey : false,
|
||||||
outSourceMap : null,
|
outSourceMap : null,
|
||||||
|
outFileName : null,
|
||||||
sourceRoot : null,
|
sourceRoot : null,
|
||||||
inSourceMap : null,
|
inSourceMap : null,
|
||||||
|
sourceMapUrl : null,
|
||||||
|
sourceMapInline : false,
|
||||||
fromString : false,
|
fromString : false,
|
||||||
warnings : false,
|
warnings : false,
|
||||||
mangle : {},
|
mangle : {},
|
||||||
@@ -54,25 +66,49 @@ exports.minify = function(files, options) {
|
|||||||
});
|
});
|
||||||
UglifyJS.base54.reset();
|
UglifyJS.base54.reset();
|
||||||
|
|
||||||
|
var inMap = options.inSourceMap;
|
||||||
|
if (typeof inMap == "string" && inMap != "inline") {
|
||||||
|
inMap = JSON.parse(fs.readFileSync(inMap, "utf8"));
|
||||||
|
}
|
||||||
|
|
||||||
// 1. parse
|
// 1. parse
|
||||||
var toplevel = null,
|
var toplevel = null,
|
||||||
sourcesContent = {};
|
sourcesContent = {};
|
||||||
|
|
||||||
if (options.spidermonkey) {
|
if (options.spidermonkey) {
|
||||||
|
if (inMap == "inline") {
|
||||||
|
throw new Error("inline source map only works with built-in parser");
|
||||||
|
}
|
||||||
toplevel = UglifyJS.AST_Node.from_mozilla_ast(files);
|
toplevel = UglifyJS.AST_Node.from_mozilla_ast(files);
|
||||||
} else {
|
} else {
|
||||||
if (typeof files == "string")
|
function addFile(file, fileUrl) {
|
||||||
files = [ files ];
|
|
||||||
files.forEach(function(file, i){
|
|
||||||
var code = options.fromString
|
var code = options.fromString
|
||||||
? file
|
? file
|
||||||
: fs.readFileSync(file, "utf8");
|
: fs.readFileSync(file, "utf8");
|
||||||
sourcesContent[file] = code;
|
if (inMap == "inline") {
|
||||||
|
inMap = read_source_map(code);
|
||||||
|
}
|
||||||
|
sourcesContent[fileUrl] = code;
|
||||||
toplevel = UglifyJS.parse(code, {
|
toplevel = UglifyJS.parse(code, {
|
||||||
filename: options.fromString ? i : file,
|
filename: fileUrl,
|
||||||
toplevel: toplevel,
|
toplevel: toplevel,
|
||||||
bare_returns: options.parse ? options.parse.bare_returns : undefined
|
bare_returns: options.parse ? options.parse.bare_returns : undefined
|
||||||
});
|
});
|
||||||
|
}
|
||||||
|
if (!options.fromString) {
|
||||||
|
files = UglifyJS.simple_glob(files);
|
||||||
|
if (inMap == "inline" && files.length > 1) {
|
||||||
|
throw new Error("inline source map only works with singular input");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
[].concat(files).forEach(function (files, i) {
|
||||||
|
if (typeof files === 'string') {
|
||||||
|
addFile(files, options.fromString ? i : files);
|
||||||
|
} else {
|
||||||
|
for (var fileUrl in files) {
|
||||||
|
addFile(files[fileUrl], fileUrl);
|
||||||
|
}
|
||||||
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
if (options.wrap) {
|
if (options.wrap) {
|
||||||
@@ -83,7 +119,7 @@ exports.minify = function(files, options) {
|
|||||||
if (options.compress) {
|
if (options.compress) {
|
||||||
var compress = { warnings: options.warnings };
|
var compress = { warnings: options.warnings };
|
||||||
UglifyJS.merge(compress, options.compress);
|
UglifyJS.merge(compress, options.compress);
|
||||||
toplevel.figure_out_scope();
|
toplevel.figure_out_scope(options.mangle);
|
||||||
var sq = UglifyJS.Compressor(compress);
|
var sq = UglifyJS.Compressor(compress);
|
||||||
toplevel = sq.compress(toplevel);
|
toplevel = sq.compress(toplevel);
|
||||||
}
|
}
|
||||||
@@ -103,14 +139,11 @@ exports.minify = function(files, options) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// 5. output
|
// 5. output
|
||||||
var inMap = options.inSourceMap;
|
var output = { max_line_len: 32000 };
|
||||||
var output = {};
|
if (options.outSourceMap || options.sourceMapInline) {
|
||||||
if (typeof options.inSourceMap == "string") {
|
|
||||||
inMap = fs.readFileSync(options.inSourceMap, "utf8");
|
|
||||||
}
|
|
||||||
if (options.outSourceMap) {
|
|
||||||
output.source_map = UglifyJS.SourceMap({
|
output.source_map = UglifyJS.SourceMap({
|
||||||
file: options.outSourceMap,
|
// prefer outFileName, otherwise use outSourceMap without .map suffix
|
||||||
|
file: options.outFileName || (typeof options.outSourceMap === 'string' ? options.outSourceMap.replace(/\.map$/i, '') : null),
|
||||||
orig: inMap,
|
orig: inMap,
|
||||||
root: options.sourceRoot
|
root: options.sourceRoot
|
||||||
});
|
});
|
||||||
@@ -129,15 +162,19 @@ exports.minify = function(files, options) {
|
|||||||
var stream = UglifyJS.OutputStream(output);
|
var stream = UglifyJS.OutputStream(output);
|
||||||
toplevel.print(stream);
|
toplevel.print(stream);
|
||||||
|
|
||||||
if (options.outSourceMap && "string" === typeof options.outSourceMap) {
|
|
||||||
stream += "\n//# sourceMappingURL=" + options.outSourceMap;
|
|
||||||
}
|
|
||||||
|
|
||||||
var source_map = output.source_map;
|
var source_map = output.source_map;
|
||||||
if (source_map) {
|
if (source_map) {
|
||||||
source_map = source_map + "";
|
source_map = source_map + "";
|
||||||
}
|
}
|
||||||
|
|
||||||
|
var mappingUrlPrefix = "\n//# sourceMappingURL=";
|
||||||
|
if (options.sourceMapInline) {
|
||||||
|
stream += mappingUrlPrefix + "data:application/json;charset=utf-8;base64," + new Buffer(source_map).toString("base64");
|
||||||
|
} else if (options.outSourceMap && typeof options.outSourceMap === "string" && options.sourceMapUrl !== false) {
|
||||||
|
stream += mappingUrlPrefix + (typeof options.sourceMapUrl === "string" ? options.sourceMapUrl : options.outSourceMap);
|
||||||
|
}
|
||||||
|
|
||||||
return {
|
return {
|
||||||
code : stream + "",
|
code : stream + "",
|
||||||
map : source_map
|
map : source_map
|
||||||
@@ -252,3 +289,47 @@ exports.writeNameCache = function(filename, key, cache) {
|
|||||||
fs.writeFileSync(filename, JSON.stringify(data, null, 2), "utf8");
|
fs.writeFileSync(filename, JSON.stringify(data, null, 2), "utf8");
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
// A file glob function that only supports "*" and "?" wildcards in the basename.
|
||||||
|
// Example: "foo/bar/*baz??.*.js"
|
||||||
|
// Argument `glob` may be a string or an array of strings.
|
||||||
|
// Returns an array of strings. Garbage in, garbage out.
|
||||||
|
exports.simple_glob = function simple_glob(glob) {
|
||||||
|
var results = [];
|
||||||
|
if (Array.isArray(glob)) {
|
||||||
|
glob.forEach(function(elem) {
|
||||||
|
results = results.concat(simple_glob(elem));
|
||||||
|
});
|
||||||
|
return results;
|
||||||
|
}
|
||||||
|
if (glob.match(/\*|\?/)) {
|
||||||
|
var dir = path.dirname(glob);
|
||||||
|
try {
|
||||||
|
var entries = fs.readdirSync(dir);
|
||||||
|
} catch (ex) {}
|
||||||
|
if (entries) {
|
||||||
|
var pattern = "^" + (path.basename(glob)
|
||||||
|
.replace(/\(/g, "\\(")
|
||||||
|
.replace(/\)/g, "\\)")
|
||||||
|
.replace(/\{/g, "\\{")
|
||||||
|
.replace(/\}/g, "\\}")
|
||||||
|
.replace(/\[/g, "\\[")
|
||||||
|
.replace(/\]/g, "\\]")
|
||||||
|
.replace(/\+/g, "\\+")
|
||||||
|
.replace(/\^/g, "\\^")
|
||||||
|
.replace(/\$/g, "\\$")
|
||||||
|
.replace(/\*/g, "[^/\\\\]*")
|
||||||
|
.replace(/\./g, "\\.")
|
||||||
|
.replace(/\?/g, ".")) + "$";
|
||||||
|
var mod = process.platform === "win32" ? "i" : "";
|
||||||
|
var rx = new RegExp(pattern, mod);
|
||||||
|
for (var i in entries) {
|
||||||
|
if (rx.test(entries[i]))
|
||||||
|
results.push(dir + "/" + entries[i]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (results.length === 0)
|
||||||
|
results = [ glob ];
|
||||||
|
return results;
|
||||||
|
};
|
||||||
|
|||||||
Reference in New Issue
Block a user