title: Profiling Node.js
author: Dave Pacheco
date: Wed Apr 25 2012 13:48:58 GMT-0700 (PDT)
status: publish
category: Uncategorized
slug: profiling-node-js
It's incredibly easy to visualize where your Node program spends its time using DTrace and node-stackvis (a Node port of Brendan Gregg's FlameGraph tool):
- Run your Node.js program as usual.
- In another terminal, run:
$ dtrace -o stacks.out -n 'profile-97/execname == "node" && arg1/{
@[jstack(100, 8000)] = count(); } tick-60s { exit(0); }'
This will sample about 100 times per second for 60 seconds and emit results to stacks.out. Note that this will sample all running programs called "node". If you want a specific process, replace execname == "node"
with pid == 12345
(the process id).
- Use the "stackvis" tool to transform this directly into a flame graph. First, install it:
$ npm install -g stackvis
then use stackvis
to convert the DTrace output to a flamegraph:
$ stackvis dtrace flamegraph-svg < stacks.out > stacks.svg
- Open stacks.svg in your favorite browser.
You'll be looking at something like this:
This is a visualization of all of the profiled call stacks. This example is from the "hello world" HTTP server on the Node.js home page under load. Start at the bottom, where you have "main", which is present in most Node stacks because Node spends most on-CPU time in the main thread. Above each row, you have the functions called by the frame beneath it. As you move up, you'll see actual JavaScript function names. The boxes in each row are not in chronological order, but their width indicates how much time was spent there. When you hover over each box, you can see exactly what percentage of time is spent in each function. This lets you see at a glance where your program spends its time.
That's the summary. There are a few prerequisites:
- You must gather data on a system that supports DTrace with the Node.js ustack helper. For now, this pretty much means illumos-based systems like SmartOS, including the Joyent Cloud. MacOS users: OS X supports DTrace, but not ustack helpers. The way to get this changed is to contact your Apple developer liason (if you're lucky enough to have one) or file a bug report at bugreport.apple.com. I'd suggest referencing existing bugs 5273057 and 11206497. More bugs filed (even if closed as dups) show more interest and make it more likely Apple will choose to fix this.
- You must be on 32-bit Node.js 0.6.7 or later, built
--with-dtrace
. The helper doesn't work with 64-bit Node yet. On illumos (including SmartOS), development releases (the 0.7.x train) include DTrace support by default.
There are a few other notes:
For more on the underlying pieces, see my previous post on Node.js profiling and Brendan's post on Flame Graphs.
Dave Pacheco blogs at dtrace.org