September 20, 2016 - by Wyatt Lyon Preul
The modular debugger (MDB) is a vital tool to have at your disposal for debugging issues in your Node.js application, no matter if they are trivial or complex, like that of the infamous Node.js memory leak. Unfortunately, setting up an environment to run MDB is often too much of a hurdle for developers to consider even using it. This post explains how to run MDB as a service to make debugging Node.js applications easier.
Over the years, Joyent has shared information on how to use MDB to debug core dumps generated from Node.js applications. The information in those posts is still valid, but there are new dcmds
that are worth a demonstration. This post demonstrates some of the new commands. Additionally, it assumes that you don't have and don't plan to setup SmartOS. The only prerequisite is that you sign up for a free Manta account and install the Manta CLI.
To use MDB with a Node.js core dump, you will need to pass in the Node.js binary used to run the process that generated the core dump. And, of course, you will also need the core dump file itself. Once you have these files saved locally, you can upload them into Manta using the mput
command. Below is an example of uploading the Node.js binary and a core dump to Manta.
mput -f node ~~/stor/
mput -f core.dump ~~/stor/
In addition to the above two files, you will also need the mdb_v8
binary file. Fortunately, this file is in Manta in a public location. As a result, you can use mln
to create a SnapLink to the file. Below is an example of setting up a SnapLink named v8_amd64.so
from the public file located at /Joyent_Dev/public/mdb_v8/v1.1.3/mdb_v8_amd64.so
.
mln /Joyent_Dev/public/mdb_v8/v1.1.3/mdb_v8_amd64.so ~~/stor/v8_amd64.so
Now that all of the files required are in Manta, you can create a new interactive session and start debugging with MDB. To create a new session use the mlogin
command specifying the two assets along with the main core file. Below is an example of running this command.
mlogin ~~/stor/core.dump -s ~~/stor/v8_amd64.so -s ~~/stor/node
mlogin
connects you to Manta with the v8_amd64.so
and node
files available in the /assets/{username}/stor/
folder. Next, load MDB by passing in the path to the Node.js binary and the core dump file, below is what this command should look like, replacing {username}
with your Manta username.
mdb /assets/{username}/stor/node $MANTA_INPUT_FILE
Load the v8_amd64.so
file into MDB in order to gain access to the Node.js dcmds
using the ::load
command.
> ::load /assets/{username}/stor/v8_amd64.so
Everything should be loaded and ready. To verify that this is the case and to see some basic information about the process that created the core dump run the ::status
command.
> ::status
debugging core file of node (64-bit) from smartos
file: /assets/{username}/stor/node
initial argv: node --abort-on-uncaught-exception server.js
threading model: raw lwps
status: process terminated by SIGILL (Illegal Instruction), addr=f401e9
dcmds
To see a listing of all of the available dcmds
, execute ::dcmds
. If you want to see more information on a particular dcmd
run ::help
followed by the command. For example, to get the help information on the ::nodebuffer
command execute the following.
> ::help nodebuffer
NAME
nodebuffer - print details about the given Node Buffer
SYNOPSIS
addr ::nodebuffer [-a]
ATTRIBUTES
Target: proc
Module: v8_amd64
Interface Stability: Unstable
For the most part, ::findjsobjects
is the command to become the most familiar with, as it will be what you will use to look for the JavaScript objects in memory. For example, if you want to see all of the requests to your process at the time the core dump was created you can use ::findjsobjects
with the constructor flag as shown below.
> ::findjsobjects -c IncomingMessage
The ::findjsobjects
command executed above, outputs the address for any object that has the constructor IncomingMessage
. You can pipe this result to ::jsprint
to see what the object values are.
> ::findjsobjects -c IncomingMessage | ::jsprint
{
"_readableState": {
"objectMode": false,
"highWaterMark": 16384,
"buffer": [...],
To see a listing of the functions that were in memory, use the ::jsfunctions
command, as shown below.
> ::jsfunctions
FUNC #FUNCS NAME FROM
378381117f51 1 (as req.once) /server.js line 11
378381117be1 1 (as req.on) /server.js line 7
378381116091 1 resOnFinish _http_server.js position 15138
If you are curious about what the actual source code of a function is, you can use the ::jssource
command. For example, to print the source of resOnFinish
shown above, the command would be the following.
> 378381116091::jssource
file: _http_server.js
491 }
492
493 // When we're finished writing the response, check if this is the last
494 // response, if so destroy the socket.
495 res.on('finish', resOnFinish);
496 function resOnFinish() {
497 // Usually the first incoming element should be our request. it may
498 // be that in the case abortIncoming() was called that the incoming
499 // array will be empty.
500 assert(incoming.length === 0 || incoming[0] === req);
Similarly, to know what variables a function has a reference to, use the ::jsclosures
command, as shown below.
> 378381116091::jsclosure
"req": 3783811141d9: {
"_readableState": 3783811142c9: [...],
"readable": 1e40ce504299: false,
"domain": 1e40ce504101: null,
When finished using MDB execute the ::quit
command and exit the Manta session using exit
.
In addition to running MDB directly, there is a utility named Thoth that can be used to upload and analyze a core dump by using Manta Functions. Use npm to install manta-thoth globally (npm -g manta-thoth
). Once installed, you can use Thoth to upload and issue commands into MDB. For example, the following shows how to upload a core into Manta and retrieve basic information about it, similar to running the ::status
command previously.
$ thoth upload core.dump
thoth: creating 2bcc33c390eda0f5b106041f3bdb5910
...
$ thoth info 2bcc33c390eda0f5b106041f3bdb5910
{
"name": "/{username}/stor/thoth/2bcc33c390eda0f5b106041f3bdb5910",
"dump": "/{username}/stor/thoth/2bcc33c390eda0f5b106041f3bdb5910/core.dump",
"pid": "50187",
"cmd": "node",
"psargs": "node --abort-on-uncaught-exception server.js",
"platform": "joyent_20160622T234741Z",
"node": "smartos",
"version": "1",
"time": 1473781733,
"stack": []
}
It should be clear that there are many powerful commands available in MDB when using the mdb_v8 binaries. Over the years many commands have been added as well as support for debugging Linux core dumps. Additionally, there isn't even a need setup a local or hosted instance of SmartOS, as Manta is all you need for debugging purposes. Therefore, there hasn't been a better time to use MDB with Manta for post-mortem Node.js debugging.