The project is a part of IntelliJ product line, spy-js is bundled with WebStorm and IndelliJ IDEA Ultimate Edition and available as a free downloadable plugin for PhpStorm, RubyMine and PyCharm Professional Edition. You can watch 7 min screencast here and read detailed feature walkthrough in WebStorm blog.
For full documentation please also visit WebStorm web help.
This documentation covers various topics not covered (or covered briefly) in WebStorm web help, and contains:
- Overview
- License
- Installation notes
- Configuration documentation
- PhantomJs
- [Karma Runner] (#karma-runner)
- Development proxy (like Fiddler or Charles proxy) configiration
- Mobile device proxy configuration
- Virtual machine configuration
- Troubleshooting, known issues and limitations
If something described in the documentation doesn't work for you, please check if it's one of the known issues. Feel free to ask your questions on stackoverflow with spy-js tag or in the repository issues.
In a nutshell, spy-js is a tool for JavaScript developers that allows to simply debug/trace/profile JavaScript on running on different platforms/browsers/devices. It fills gaps that existing browser development tools have and tackles common development tasks from a different angle.
Documentation and code samples in this repository are licensed under MIT. Spy-js tool itself isn’t open source.
To use WebStorm integration of spy-js: install latest WebStorm version. To use spy-js in PhpStorm, RubyMine and PyCharm Professional Edition: install this plugin from JetBrains plugin repository.
Spy-js can work with zero configuration or be configured with *.conf.js configuration file.
Configuration code is just a simple valid JavaScript file.
module.exports = function ($) {
// $.configSetting1 = value1;
// $.configSetting2 = value2;
// many other settings
// ...
};
In order to configure your session you use session configuration parameter $
passed in and assign its supported properties desired values. If you've already used grunt, you'll find the process very similar.
Root URL is a URL of the page/website you'd like to trace. Set it to isolate your session from receiving events from any other websites/pages.
$.root = 'http://localhost:3002/';
URL mapper is one of the most important bits of the session configuration. By given URL of the script, mapper function is supposed to return configuration object for the script. The configuration object contains various settings affecting how the script should be processed by spy-js.
Mapper function can return different configuration objects for different script URLs. Configuration object should have following structure:
// configuration object
{
instrument: true (default instrumentation settings used)
| false (script will not be traced)
| instrumentation settings object | not set (same as false)
}
// instrumentation settings object
{
prettify: true | false (default) | not set (same as false),
objectDump: false | dump settings object
| not set (default, dump settings object with default settings used)
}
// dump settings object
{
depth: number | not set (default 1 used),
propertyNumber: number | not set (default 3 used),
arrayLength: number | not set (default 3 used),
stringLength: number | not set (default 50 used)
}
instrument
property specifies how the script from given URL will be modified (if at all) and what data will be collected in runtime. Set the property to false to not trace the script. Set the property to true to use default values or specify the object with settings.
instrument.prettify
property specifies whether to make the script look nice in case if it is minified or badly formatted.
instrument.objectDump
property specifies whether any runtime data (apart from stack trace) should be collected and to what extent. Set the property to false if you don't want to collect the runtime values. Don't set the property if you want to use default object dump settings or specify your own settings object.
instrument.objectDump.depth
property specifies how many levels of nesting within an object should be traversed and logged.
instrument.objectDump.propertyNumber
property specifies how many first properties within an object should be logged (at any allowed level of logged object hierarchy).
instrument.objectDump.arrayLength
property specifies how many array elements should be logged (at any allowed level of logged object hierarchy).
instrument.objectDump.stringLength
property specifies limit of number of characters in logged string properties.
For example, if a function in runtime returns something like
[{
a: {
p1: true
},
b: 'abcdefghij',
c: false
},
{
a : {}
}]
and instrument.objectDump.depth
is 2, instrument.objectDump.propertyNumber
is 2, instrument.objectDump.arrayLength
is 1 and instrument.objectDump.stringLength
is 2, then in the stack trace or the code editor you'll be able to see following representation of the object:
[
0:
{
a: {...}
b: "ab..."
...
}
1: ...
]
Because of the depth constraint, we don't see the first array element "a" property value (level 1 is traversing array, level 2 is traversing its element); only see two first properties of the first array element because of the property number constraint; only see the first two characters of the first array element "b" string property because of the string length constraint; only see the first array element object dump because of the array length constraint.
Event filter allows to specify what kind of events are captured and displayed in spy-js UI. There are five properties in the event filter object that you can set: globalScope, timeout, interval, events and noEvents.
With the following event filter, spy-js will show program execution scope (scripts loading), intervals, will not show timeouts, will show all events except blacklisted DOMContentLoaded and keyup.
$.eventFilter = {
globalScope: true,
timeout: false,
interval: true,
noEvents: ['DOMContentLoaded', 'keyup']
};
With the following event filter, spy-js will not show program execution scope (scripts loading), will show intervals and timeouts, will only show whitelisted click and keyup events.
$.eventFilter = {
globalScope: false,
timeout: true,
interval: true,
events: ['click', 'keyup']
};
Let's have a look at the full sample of a configuration file to see how it all works together.
module.exports = function ($) {
$.root = 'http://localhost:3002/';
$.mapper = function (url) {
if (url.indexOf('jquery') >= 0) {
return {
instrument: false
};
}
if (url.indexOf('underscore-1.4.4.min.js') >= 0) {
return {
instrument: {
prettify: true,
objectDump: false
}
};
}
return {
instrument: {
prettify: false,
objectDump: {
depth: 1,
propertyNumber: 3,
arrayLength: 3,
stringLength: 50
}
}
};
};
$.eventFilter = {
globalScope: false,
timeout: true,
interval: true,
events: ['click']
};
};
The sample assumes that you have a locally hosted web project at http://localhost:3002/
.
Mapper above is configured not to trace jQuery (or any jQuery plugins or related scripts if they have "jquery" string in their URL). It is also configured to trace minified underscore library, prettify it, but not collect any runtime data from it.
Mapper is not prettifying them because these are local scripts that are formatted nicely anyway and constraining object dumps to reasonably low limits to avoid any performance issues on the traced website (traversing objects and sending dumps across the wire does take additional time).
When tracing the website, according to the specified event filter we'll only see the code executed by timeouts, intervals and click events.
For best performance
- opt out prettifying already nicely formatted non-minified files
- set reasonably low object dump limits
- exclude scripts you are not interested in by setting instrument to false. This can be done for example for libraries you wouldn't want to step in
- whitelist/blacklist events you would like/would not like to see by using event filter setting
It is recommended to save your session configuration file as spy.conf.js (or spy-all.conf.js/spy-nolibs.conf.js etc.) in your project folder and commit/check in your VCS system so the configuration could be shared across your project team.
PhantomJS is a headless WebKit, so simple spy-js configuration is sufficient to trace web pages loaded by it.
Please note, that because of PhantomJs limitation, when you'd like to trace localhost, you should use your machine name instead.
In the example below I'll illustrate how to use spy-js to trace Jasmine tests running by PhantomJs from Grunt (using WebStorm Grunt console).
Create a new spy-js run configuration as on the screenshot below. Start the configuration.
Create Gruntfile.js and Jasmine spec as illustrated below. Note that machine name used as a host and host name. Run default grunt task.
Switch to spy-js tool window to work with the trace.
I'm also using following capture exclusion to avoid noise from jasmine task generated files.
In the example below I'll illustrate how to use spy-js to trace Jasmine tests running from Karma runner in Chrome and PhantomJs.
Create a new spy-js run configuration as on the screenshot below. Start the configuration.
Create karma.conf.js and Jasmine spec as illustrated below. Note that machine name used as a host name in order for PhantomJs to work.
Create and launch karma run configuration.
Wait till your test suite execution is finished.
Switch to spy-js tool window to work with the trace.
I'm also using following capture exclusion to avoid noise from library files.
If you're using development proxy, for instance to map minified files and replace them with local development versions, it is still possible to use spy-js to instrument and trace those development versions of JavaScript files.
In the example below it's illustrated how to make Fiddler work with spy-js in WebStorm. The idea is to request your non-minified files via spy-js proxy server using local proxy mode URLs. In the example I'll replace minified sh_javascript.min.js file from nodejs.org with my version sh_javascript.js using Fiddler and will trace sh_javascript.js execution using spy-js.
Create a new spy-js run configuration, uncheck "Automatically configure system proxy" checkbox.
Configure AutoResponder in Fiddler as follows:
Start Fiddler and spy-js run configuration (the order doesn't matter), access the page in browser and see it traced in WebStorm.
Note that localhost:3546 is spy-js trace server launched by WebStorm, localhost:8080 is an address where your local non-minified files are hosted. I have used static http-server to host them, but you can use any web server.
First AutoResponder rule uses a regular expression to replace all files like file.min.js with corresponding file.js, last two rules are mandatory to make spy-js work on the page.
Even though spy-js itself doesn't support tracing HTTPS secured web pages, you can still use it for tracing with a tool that can do SSL termination, for example with a development proxy like Fiddler or Charles Proxy.
In the example below it's illustrated how to make Fiddler work with spy-js in WebStorm to trace HTTPS secured website. The idea is to use Fiddler to decrypt HTTPS traffic and request JavaScript files via spy-js proxy server using local proxy mode URLs. In the example I'll demonstrate how to set up Fiddler and spy-js to trace GitHub website. Other development proxies may need a bit different configuration, but the idea should still be applicable.
Create a new spy-js run configuration, uncheck "Automatically configure system proxy" checkbox.
Configure Fiddler to decrypt HTTPS traffic:
Configure AutoResponder in Fiddler as follows:
First AutoResponder rule uses a regular expression to request all *.js files via spy-js proxy, last two rules are mandatory to make spy-js work on the page.
Modify Fiddler rules (Rules - Customize Rules) as follows:
Highlighted line just adds an additional HTTP response header, the header is required to configure Content Security Policy (for browsers that support it) to evaluate spy-js tracer code.
Start Fiddler and spy-js run configuration (the order doesn't matter), access some HTTPS secured page in browser and see it traced in WebStorm.
See configuration example for iOS in spy-js WebStorm blog post
If you are using virtual machine, for instance you have a develepment machine with Mac and WebStorm spy-js installed and hosting Windows VM via Virtual Box on it, it is possible to use spy-js to trace web sites opened inside IE (or another browser) on the VM instance and view the trace in your WebStorm.
First, create and start spy-js run configuration in WebStorm on your development machine. You can use default trace server port (3546), leave "URL to trace" field empty and do not check "Automatically configure system proxy" checkbox (because the proxy configuration is not required on your development machine, it's required on the VM and will be performed manually).
Second, you need to configure your VM system proxy, for example see how to do it on Windows. The settings you'll need to specify are:
- IP address of your development machine for the address (avoid using fake 10.0.2.2 IP address)
- spy-js run configuration trace server port for the port (3546, if you didn't change it)
- uncheck "Bypass proxy server for local addresses" setting
- go to "Advanced" settings and add <-loopback> string to the exceptions field down the bottom
Now you can open any external website, like http://nodejs.org, in any browser on your VM and make sure it is being traced in WebStorm running on your development machine.
When you are tracing local sites (hosted on your development machine), and when you're trying to trace them they hang, you may need to access them by dev machine name or dev machine proper network IP address (instead of fake 10.0.2.2 IP address). Please note that WebStorm built-in HTTP server doesn't support such access, so you may need to install and use some other development web server for this task, for example static http-server.
To sum it up, if you have installed http-server (npm install http-server -g) and started it (by running http-server inside your application folder) using it's default port 8080, started described spy-js run configuration on your development machine (for instance named dev-box), configured your VM as described, opened a browser on the VM and accessed http://dev-box:8080, you should be able to see the the trace in your running WebStorm instance on your development machine.
If spy-js tracing doesn't work for you (and console output or log file doesn't contain any explanation):
- if you're using Chrome, when tracing session is running make sure chrome://net-internals/#proxy page "Effective proxy settings" is using spy-js-proxy.pac (if not, try re-applying the settings or clearing bad proxy list on the page if any)
- do force refesh (
Ctrl/Command + F5
orCtrl/Command + R
) on the traced page to make sure the traced website scripts are not cached in your browser - tracing scripts with incorrect (or without) Content-Type response header is not supported. Some development web servers have default settings with incorrect Content-Type response header (or no Content-Type response header) for JavaScript files. Use your browser dev tools to check whether your script has correct Content-Type response header:
text/javascript
orapplication/x-javascript
orapplication/javascript
. - tracing https secure websites is not supported at the moment, however it is possible to set up a development proxy to use with spy-js to trace HTTPS secured pages
- tracing HTML pages inline JavaScript is not supported at the moment
- integrated windows authentication is not supported
- execution time figures display the time required to execute modified code and thus may significantly differ from the real figures; relatively to other spy-js collected performance metrics they make sense and may be used to identify and fix bottlenecks etc.; use target browser built-in profiler tools/v8 tools for node.js for precise figures