Experimental tools for debugging Node.js apps without pausing.
Just like with a debugger, these are not meant to run in production in most cases.
Tools added via simple require
node --require=TOOL app.js
or in the beginning of your program:
require('TOOL')
All outputs from debugging-aid start with [aid]
[aid] network, outgoing
to: http://127.0.0.1:33453/something?attr=value
stack: at Agent.createSocket (_http_agent.js:234:26)
at Agent.addRequest (_http_agent.js:193:10)
...
name to require | description |
---|---|
debugging-aid/promises |
Supports basic handlers for detecting potential issues with promises. Logs them to console |
debugging-aid/blocked |
Attempt to list stack traces for functions synchronously blocking the event loop |
debugging-aid/leak |
Detects a potential memory leak by noticing memory consumprion growing across 5 GC passes |
debugging-aid/network |
Lists all outgoing network requests with their full URL and a stack trace pointing to the code making the request |
debugging-aid/curl |
Lists all outgoing http requests as curl commands |
debugging-aid/moduse |
Lists stack traces pointing to where a module is required. Module name is passed via AID_MODULE env variable - see details below |
debugging-aid/promisecount |
prints the number of promises created while the process was running right before it exits |
debugging-aid/hooks |
Produces a tree of asynchronous calls and perf hooks output containing a diagram of asynchronous calls with their timing in the application run. See instructions below |
Tested in Node.js v12
In newer versions of Node, some of the internal modules are using ES Modules and imports. Hooks for that exist, but are still very experimental. I might add them, but for now be aware this tool doesn't work with imports. https://nodejs.org/api/esm.html#esm_experimental_loaders
To list traces to all modules that require the net
internal module:
AID_MODULE=net node --require debugging-aid/moduse app.js
To list traces to where a non-internal module is required, pass an absolute path to a file.
This would list where request
was required, but only the one installed top-level
AID_MODULE='/home/you/app/node_modules/request/index.js' node --require debugging-aid/moduse app.js
AID_HOOK_SCOPE='what to look for' node --trace-event-categories node.perf --require=debugging-aid/hooks app.js
The tool will look for AID_HOOK_SCOPE in stack traces of all functions called in new asynchronous context and trace their execution along with their descendants.
- set AID_HOOK_SCOPE to the path to a file with code you're interested in tracing. (:linenumber will also work, it's just searching for a string in stack traces)
- Run it in the VScode terminal and you can click links in the output to jump straight to locations in code.
- Load the node_trace file to the
about:tracing
interface in chromium/chrome dev tools and see the beginning and the ending (hence the time of synchronous execution) of each function traced.
How to interpret the output:
└[2->3] Promise.then (<anonymous>) at start (/storage/projects/github/debugging-aid/test/cases/promise.js:7:8)
- [2->3] means the asynchronous jump started at the asyncId==2 and current asyncId is 3. Using indentation hints and these numbers you can follow a chain of promises or other asynchronous jumps.
- SOMETHING1 at SOMETHING2 (path) - SOMETHING1 indicates what has been run. SOMETHING2 is the name of the function inside of which the asynchronous call was initiated. Path points to an exact line in code.
- The lines are printed after each function ends its synchronous execution
This is very experimental and might provide incorrect information under unusual circumstances.
Also, will definitely not work if your file paths contain braces. A lot of stacktrace parsing is involved.
Tested on Node.js v12.8.0
Should work in some later versions of Node.js v11 and should not work in anything older than that.
Run this to try it out on an example - should help you understand what it does.
AID_HOOK_SCOPE='test/cases/promise' node --trace-event-categories node.perf test/manual-hooks.js
AID_HOOK_SCOPE='test/cases/helper' node --trace-event-categories node.perf test/manual-hooks.js
If you're interested
- Tools use
process._rawDebug
to print to console to avoid interfering with async hooks etc. - debugging-aid/promises uses
multipleResolves
andunhandledRejection
events - debugging-aid/network hooks into socket implementation, so it can only print the URL, won't print http method etc. (unless you decide to contribute that)
- debugging-aid/moduse hooks into node's internal module cache and checks if there's been a cache fetch for the name of the module
- get
moduse
to also support imports, not just require - pad asyncIDs with zeros in hooks so they're displayed correctly in chrome://tracing