DTrace Your Node.js JS
Mark Cavage
dtrace.conf April 03, 2012
Me
- @mcavage (Twitter)
- mcavage (GitHub)
- mcavage at gmail dot com
Just to level-set (i.e., What's Node.js?)
data:image/s3,"s3://crabby-images/b9ba9/b9ba9de11e790c9ad1504f5e937794508ad90dea" alt=""
What's Node.js?
- Server-side Javascript.
- Always non-blocking (threads aren't exposed to you)
- And networking/HTTP are first class
Why would you use Node.js?
- You want to spin up some new network service (relatively) quickly
- You want it to talk to other things
- And you care about performance/scalability
Bored now. What's this talk about?
- (Briefly) What DTrace is available for node itself
- How to DTrace your application with node-dtrace-provider
- Some example usage of dtrace-provider (ldapjs/restify)
DTrace the Node core
- Basically, you've got USDT probes for socket, http and gc
- Those let you get the basics of what your app is doing
- But it's a pretty course granularity
Some One-liners (thanks @bmc)
Cool - do I need anything else?
- As said before, those probes tell you a lot about what the interpreter sees
- They don't tell you much about your app's components
- Wouldn't you like to know if say a DB call or a FS call were what was slowing your app down?
What if your infrastructure looks like this?
data:image/s3,"s3://crabby-images/0bffe/0bffeda9c0f6cf406d743b650787c8a3744408b4" alt=""
Suppose you had some (crappy) REST api...
Wouldn't it be nice to see that slowdown?
In action?
Wait. What?
- There was some magic in restify
- Under the covers it creates DTrace probes for all the handlers in a server
- The rest of this talk is about showing you how to do that
Thanks Chris Andrews!
$ npm install dtrace-provider
- Allows you to DTrace for your Node.js application code
- Installs everywhere; useful on Solaris derivations/OSX
- You don't need to be root
What is node-dtrace-provider?
Is this magic? How does this work?
- Dynamically create tracepoints in the running process
- Dynamically creates DOF to describe the provider to the kernel
- ioctl said DOF into the kernel
- For more information, see Chris' blog
How do you use it?
Node's read vs system read
Pretty decent in terms of interpreter overhead (μs)
(Tangent) What's with the goofy read:return?
- man -s 2 read says if (read(fd) == 0) then EOF
- node however decides to keep calling it. and calling it. and calling it.
- Hence, that janky D script to get the timing of the first return 0 (since the system is ready to move on)
Bah! Show me some examples!
ldapjs: hello world server
Show me the probes!
Some search one(ish) liners
Fancy: Visualization of ldapjs
data:image/s3,"s3://crabby-images/13cc3/13cc34689dec23c8d24ca601926f51d855c443d3" alt=""
That said: it's pretty course granularity
- This was my first pass at integrating dtrace into node app code.
- Really there's just start/done for each LDAP operation
- Lets us see what a server is doing, but not very well broken out
Take two: restify
- Restify lets you build a REST api by pipelining handlers
- After having done ldapjs, what I really wanted was to be able to visualize all the steps in a request
- DTrace start/done probes get inserted for every distinct handler
A (slightly) bigger exampe
Now we're talking!
The same D script as before...
Shows us a lot more data
Every time a route/handler is added
- restify sets up a start/done DTrace probe for the route
- plus a start/done probe for each handler in the route
- plus a start/done probe for the shared handlers to track in that route
- You can glob to your heart's content in D scripts, since the names are predictable
mapping code
Route runner fires
Some tips on how to use dtrace-provider...
data:image/s3,"s3://crabby-images/7d445/7d44556f613af7a00592bf2aa9274c2a86144247" alt=""
Ride on top of other probes
data:image/s3,"s3://crabby-images/2e1fa/2e1fae8e7788410a3af5edc74515cee3a44db035" alt=""
Whenever possible, piggyback on system probes by using an argument you can correlate (like filename in the nodecat example)
Picking arguments
data:image/s3,"s3://crabby-images/8d48a/8d48a43e97fd636a70bca58ce81f781a5eb46e4b" alt=""
There's no support for argument structs (yet), and you only get 6 arguments to play with, so choose your args wisely
Build into your framework
data:image/s3,"s3://crabby-images/8434d/8434dba0d01ea670be4cfc8ce25bf8db40ad851a" alt=""
ldapjs and restify are two examples where DTrace isn't in user code at all, but provides substantial value.
Easy latency tracking
- For any given probe, add -start and -done so latency can be calculated
- Where it makes sense, make arg0 be a monotonically increasing integer
What would be nice in the future?
- JS objects as probe arguments (i.e., structs)
- Being able to tack on more probes on the fly (i.e., after calling enable())
- Minimizing the disabled probe effect
And with that...
data:image/s3,"s3://crabby-images/1f35d/1f35d16e8ba973e7795d21b50ef702e78452c11f" alt=""
Questions?
data:image/s3,"s3://crabby-images/a8abe/a8abe49f07a82f5dac949e03d56a507473ca5d0a" alt=""
/
#