New Case Study:See how Anthropic automated 95% of dependency reviews with Socket.Learn More
Socket
Sign inDemoInstall
Socket

optimize-js

Package Overview
Dependencies
Maintainers
1
Versions
4
Alerts
File Explorer

Advanced tools

Socket logo

Install Socket

Detect and block malicious and high-risk dependencies

Install

optimize-js

Optimize initial JavaScript execution/parsing by wrapping eager functions

  • 1.0.2
  • Source
  • npm
  • Socket score

Version published
Weekly downloads
2.6K
decreased by-16.4%
Maintainers
1
Weekly downloads
 
Created
Source

optimize-js Build Status JavaScript Style Guide

Optimize a JavaScript file for faster initial execution and parsing, by wrapping all immediately-invoked functions or likely-to-be-invoked functions in parentheses.

Install

npm install -g optimize-js

Usage

optimize-js input.js > output.js

Example input:

!function (){}()
function runIt(fun){ fun() }
runIt(function (){})

Example output:

!(function (){})()
function runIt(fun){ fun() }
runIt((function (){}))

Benchmark overview

BrowserTypical speed boost using optimize-js
Chrome 5257.09%
Edge 1428.88%
Firefox 4812.49%
Safari 106.54%

For benchmark details, see benchmarks.

CLI

Usage: optimize-js [ options ]

Options:
  --source-map  include source map                                     [boolean]
  -h, --help    Show help                                              [boolean]

Examples:
  optimize-js input.js > output.js    optimize input.js
  optimize-js < input.js > output.js  read from stdin, write to stdout

JavaScript API

var optimizeJs = require('optimize-js');
var input = "!function() {console.log('wrap me!')}";
var output = optimizeJs(input); // "!(function() {console.log('wrap me!')})()"

You can also pass in arguments:

var optimizeJs = require('optimize-js');
var input = "!function() {console.log('wrap me!')}";
var output = optimizeJs(input, {
  sourceMap: true
}); // now the output has source maps

Why?

Modern JavaScript engines like V8, Chakra, and SpiderMonkey have a heuristic where they pre-parse most functions before doing a full parse. The pre-parse step merely checks for syntax errors while avoiding the cost of a full parse.

This heuristic is based on the assumption that, on the average web page, most JavaScript functions are never executed or are lazily executed. So a pre-parse can prevent a slower startup time by only checking for what the browser absolutely needs to know about the function (i.e. whether it's syntactically well-formed or not).

Unfortunately this assumption breaks down in the case of immediately-invoked function expressions (IIFEs), such as these:

(function () { console.log('executed!') })();
(function () { console.log('executed Crockford-style!') }());
!function () { console.log('executed UglifyJS-style!') }();

The good news is that JS engines have a further optimization, where they try to detect such IIFEs and skip the pre-parse step. Hooray!

The bad news, though, is that these heuristics don't always work, because they're based on a greedy method of checking for a '(' token immediately to the left of the function. (The parser avoids anything more intricate because it would amount to parsing the whole thing, negating the benefit of the pre-parse). In cases without the paren (which include common module formats like UMD/Browserify/Webpack/etc.), the browser will actually parse the function twice, first as a pre-parse and second as a full parse. This means that the JavaScript code runs much more slowly overall, because more time is spent parsing than needs to be. See "The cost of small modules" for an idea of how bad this can get.

Luckily, because the '(' optimization for IIFEs is so well-established, we can exploit this during our build process by parsing the entire JavaScript file in advance (a luxury the browser can't afford) and inserting parentheses in the cases where we know the function will be immediately executed (or where we have a good hunch). That's what optimize-js does.

More details on the IIFE optimization can be found in this discussion. Some of my thoughts on the virtues of compile-time optimizations can be found in this post.

FAQs

How does it work?

The current implementation is to parse to a syntax tree and check for functions that:

  1. Are immediately-invoked via any kind of call statement (function(){}(), !function(){}(), etc.)
  2. Are passed in directly as arguments to another function

The first method is an easy win – those functions are immediately executed. The second method is more of a heuristic, but tends to be a safe bet given common patterns like Node-style errbacks, Promise chains, and UMD/Browserify/Webpack module declarations.

In all such cases, optimize-js wraps the function in parentheses.

But... you're adding bytes!

Yes, optimize-js might add as many as two bytes (horror!) per function, which amounts to practically nil once you take gzip into account. To prove it, here are the gzipped sizes for the libraries I use in the benchmark:

ScriptSize (bytes)Difference (bytes)
benchmarks/ember.min.js126957
benchmarks/ember.min.optimized.js127134+177
benchmarks/immutable.min.js15782
benchmarks/immutable.min.optimized.js15809+27
benchmarks/jquery.min.js30450
benchmarks/jquery.min.optimized.js30524+74
benchmarks/lodash.min.js24528
benchmarks/lodash.min.optimized.js24577+49
benchmarks/pouchdb.min.js45298
benchmarks/pouchdb.min.optimized.js45426+128
benchmarks/three.min.js125070
benchmarks/three.min.optimized.js125129+59

Is optimize-js intended for library authors?

Sure! If you are already shipping a bundled, minified version of your library, then there's no reason not to apply optimize-js (assuming you benchmark your code to ensure it does indeed help!). However, note that optimize-js should run after Uglify, since Uglify strips extra parentheses and also negates IIFEs by default. This also means that if your users apply Uglification to your bundle, then the optimization will be undone.

Also note that because optimize-js optimizes for some patterns that are based on heuristics rather than known eagerly-invoked functions, it may actually hurt your performance in some cases. (See benchmarks below for examples.) Be sure to check that optimize-js is a help rather than a hindrance for your particular codebase, using something like:

var start = performance.now();
// your code goes here...
var end = performance.now();
console.log('took ' + (end - start) + 'ms');

Also, be sure to test in multiple browsers! If you need an Edge VM, check out edge.ms.

Shouldn't this be Uglify's job?

Possibly! This is a free and open-source library, so I encourage anybody to borrow the code or the good ideas. :)

Why not paren-wrap every single function?

As described above, the pre-parsing optimization in browsers is a very good idea for the vast majority of the web, where most functions aren't immediately executed. However, since optimize-js knows when your functions are immediately executed (or can make reasonable guesses), it can be more judicious in applying the paren hack.

Does this really work for every JavaScript engine?

Based on my tests, this optimization seems to work best for V8 (Chrome), followed by Chakra (Edge), followed by SpiderMonkey (Firefox). For JavaScriptCore (Safari) it seems to be basically a wash, although it's hard to tell because JSCore is so fast already that the numbers are tiny.

In the case of Chakra, Uglify-style IIFEs are actually already optimized, but using optimize-js doesn't hurt because a function preceded by '(' still goes into the fast path.

Benchmarks

These tests were run using a handful of popular libraries, wrapped in performance.now() measurements. Each test reported the median of 251 runs. optimize-js commit da51013 was tested. Minification was applied using uglifyjs -mc, Uglify 2.7.0.

You can also try a live version of the benchmark (note: very slow, running locally is recommended).

Chrome 52, macOS Sierra, 2013 MacBook Pro i5

ScriptOriginalOptimizedImprovementMinifiedMin+OptimizedImprovement
ImmutableJS12.76ms1.75ms86.29%10.47ms1.68ms83.95%
jQuery26.73ms8.72ms67.38%23.02ms9ms60.90%
Lodash29.13ms28.03ms3.78%20.7ms24.45ms-18.12%
Ember1.48ms1.33ms10.14%70.93ms1.24ms98.25%
PouchDB60.98ms31.59ms48.20%40.63ms32.02ms21.19%
ThreeJS10.6ms10.16ms4.15%66.18ms10.33ms84.39%

Overall improvement: 57.09%

Edge 14, Windows 10 RS1, SurfaceBook i5

ScriptOriginalOptimizedImprovementMinifiedMin+OptimizedImprovement
ImmutableJS5.11ms1ms80.43%4.61ms1ms78.31%
jQuery21.62ms12.31ms43.06%21.27ms12.69ms40.34%
Lodash22.1ms21.81ms1.31%19.97ms19.05ms4.61%
Ember0.33ms0.32ms3.03%0.33ms0.32ms3.03%
PouchDB37.54ms28.44ms24.24%47.81ms65.23ms-36.44%
ThreeJS30.9ms19.08ms38.25%68.94ms18.26ms73.51%

Overall improvement: 28.88%

Firefox 48, macOS Sierra, 2013 MacBook Pro i5

ScriptOriginalOptimizedImprovementMinifiedMin+OptimizedImprovement
ImmutableJS4.92ms1.74ms64.63%2.87ms1.31ms54.36%
jQuery15.73ms12.64ms19.64%14.88ms12.43ms16.47%
Lodash13.8ms14.6ms-5.80%9.27ms9.33ms-0.65%
Ember3.22ms5.55ms-72.36%9.42ms5.27ms44.06%
PouchDB15.05ms16.65ms-10.63%11.96ms11.6ms3.01%
ThreeJS21.09ms17.63ms16.41%26.83ms21.68ms19.19%

Overall improvement: 12.49%

Safari 10, macOS Sierra, 2013 MacBook Pro i5

ScriptOriginalOptimizedImprovementMinifiedMin+OptimizedImprovement
ImmutableJS1.23ms1.39ms-13.01%1.22ms1.12ms8.20%
jQuery4.16ms3.91ms6.01%3.83ms3.94ms-2.87%
Lodash3.95ms3.9ms1.27%3.38ms3.36ms0.59%
Ember0.55ms0.5ms9.09%0.44ms0.4ms9.09%
PouchDB2.31ms2.15ms6.93%2.32ms2.21ms4.74%
ThreeJS8.51ms7.62ms10.46%8.03ms6.82ms15.07%

Overall improvement: 6.54%

Note that these results may vary based on your machine, how taxed your CPU is, gremlins, etc. I ran the full suite a few times on all browsers and found these numbers to be roughly representative. However, the final "overall improvement" may vary by as much as 5%, and individual libraries can swing a bit too.

Plugins

See also

Thanks

Thanks to @krisselden, @bmaurer, and @pleath for explaining these concepts in the various GitHub issues. Thanks also to astexplorer, acorn, and magic-string for making the implementation so easy.

Contributing

Build and run tests:

npm install
npm test

Run the benchmarks:

npm run benchmark # then open localhost:9090 in a browser

Test code coverage:

npm run coverage

Keywords

FAQs

Package last updated on 31 Oct 2016

Did you know?

Socket

Socket for GitHub automatically highlights issues in each pull request and monitors the health of all your open source dependencies. Discover the contents of your packages and block harmful activity before you install or update your dependencies.

Install

Related posts

SocketSocket SOC 2 Logo

Product

  • Package Alerts
  • Integrations
  • Docs
  • Pricing
  • FAQ
  • Roadmap
  • Changelog

Packages

npm

Stay in touch

Get open source security insights delivered straight into your inbox.


  • Terms
  • Privacy
  • Security

Made with ⚡️ by Socket Inc