Détail du package

0x

davidmarkclements247kMIT5.8.0

🔥 single-command flamegraph profiling 🔥

readme

0x

0x

🔥 single-command flamegraph profiling 🔥

Discover the bottlenecks and hot paths in your code, with flamegraphs.

Visualize Stack Traces

0x can profile and generate an interactive flamegraph for a Node process with a single command, on any platform which Node runs on (macOs, Linux, Windows, Android...).

Support

  • Node v12.x and above
  • Default usage supports any Operating System that Node runs on!
  • Chrome
    • Other browsers may open flamegraphs in a degraded, but functional form

Demo

An example interactive flamegraph can be viewed at http://davidmarkclements.github.io/0x-demo/

Install

npm install -g 0x

Usage

Use 0x to run a script:

0x my-app.js

Immediately open the flamegraph in the browser:

0x -o my-app.js

Automatically execute profiling command against the first port opened by profiled process:

0x -P 'autocannon localhost:$PORT' my-app.js

Use a custom node executable:

0x -- /path/to/node my-app.js

Pass custom arguments to node:

0x -- node --zero-fill-buffers my-app.js

for pwsh users, switch to CMD at first or run with npx

npx 0x -o my-app.js

Generating

When ready to generate a flamegraph, send a SIGINT or a SIGTERM.

The simplest way to do this is pressing CTRL+C.

When 0x catches the SIGINT or the SIGTERM, it process the stacks and generates a profile folder (<pid>.0x), containing flamegraph.html.

The UI

The flamegraph.html file contains the 0x UI, which is explained in docs/ui.md.

Production Servers

A lightweight, production server friendly, approach to generating a flamegraph is described in docs/production-servers.md.

The Profile Folder

By default, a Profile Folder will be created and named after the PID, e.g. 3866.0x (we can set this name manually using the --output-dir flag).

The Profile Folder is explained in more detail in docs/profile-folder.md

Example

Clone this repo, run npm i -g and from the repo root run

0x examples/rest-api

In another tab run

npm run stress-rest-example

To put some load on the rest server, once that's done use ctrl + c to kill the server.

Command Line API

--help | -h

Print usage info.

--open | -o

Open the flamegraph in the browser using open or xdg-open (see https://www.npmjs.com/package/open for details).

--on-port | -P

Run a given command and then generate the flamegraph. The command as specified has access to a $PORT variable. The $PORT variable is set according to the first port that profiled process opens.

For instance, here's an example of using autocannon to load-test the process:

0x -P 'autocannon localhost:$PORT' app.js

When the load-test completes, the profiled processed will be sent a SIGINT and the flamegraph will be automatically generated.

Remember to use single quotes to avoid bash interpolation, or else escape variable (e.g. 0x -P "autocannon localhost:$PORT" app.js won't work wheras 0x -P "autocannon localhost:\$PORT" app.js will).

Note: On Windows interpolation usually occurs with %PORT%, however in this case the dollar-prefix $PORT is the correct syntax (because the interpolation is not shell based).

Default: ''

--name

The name of the HTML file, without the .html extension Can be set to - to write HTML to STDOUT (note due to the nature of CLI argument parsing, this must be set using =, e.g. --name=-).

If either this flag or --output-html-file is set to - then the HTML will go to STDOUT.

Default: flamegraph

---title

Set the title to display in the flamegraph UI.

Default: the command that 0x ran to start the process

--output-dir | -D

Specify artifact output directory. This can be specified in template form with possible variables being {pid}, {timestamp}, {name} (based on the --name flag) and {outputDir}(variables must be specified without whitespace, e.g. { pid } is not supported).

Default: {pid}.0x

--output-html | -F

Specify destination of the generated flamegraph HTML file. This can be specified in template form with possible variables being {pid}, {timestamp}, {name} (based on the --name flag) and {outputDir} (variables must be specified without whitespace, e.g. { pid } is not supported). It can also be set to - to send the HTML output to STDOUT (note due to the nature of CLI argument parsing, this must be set using =, e.g. --output-html=-).

If either this flag or --name is set to - then the HTML will go to STDOUT.

Default: {outputDir}/{name}.html

--kernel-tracing

Use an OS kernel tracing tool (perf on Linux). This will capture native stack frames (C++ modules and Libuv I/O), but may result in missing stacks from Node.js due to the optimizing compiler.

See docs/kernel-tracing.md for more information.

Default: false

--quiet | -q

Limit output, the only output will be fatal errors or the path to the flamegraph.html upon successful generation.

Default: false

--silent | -s

Suppress all output, except fatal errors.

Default: false

--collect-only

Don't generate the flamegraph, only create the Profile Folder, with relevant outputs.

Default: false

--collect-delay

Delay the collection of stacks by a specified time(ms) relative to the first entry.

Default: 0

--visualize-only

Supply a path to a profile folder to build or rebuild visualization from original stacks.

Default: undefined

--visualize-cpu-profile

Supply a path to a CPU profile (.cpuprofile). See examples/cpu-profile for examples.

CPU Profile output does not have as much information but it can be exported from Chrome Devtools in the browser. There's also an automated headless tool for doing so: automated-chrome-profiling. For creating Node.js Cpu Profiles in Node see v8-profiler or v8-profiler-next. They can also be generated from Node.js 12 and above using the command-line flag --cpu-prof.

Default: undefined

--kernel-tracing-debug

Show output from perf(1) tools.

Default: false

--tree-debug

Save the intermediate tree representation of captured trace output to a JSON file.

Default: false

Programmatic API

0x can also be required as a Node module and scripted:

const zeroEks = require('0x')
const path = require('path')

async function capture () {
  const opts = {
    argv: [path.join(__dirname, 'my-app.js'), '--my-flag', '"value for my flag"'],
    workingDir: __dirname
  }
  try {
    const file = await zeroEks(opts)
    console.log(`flamegraph in ${file}`)
  } catch (e) {
    console.error(e)
  }
}

capture()

The Programmatic API is detailed in docs/api.md.

Troubleshooting

Memory Issues

Very complex applications with lots of stacks may hit memory issues.

The --stack-size flag can be used to set the memory to the maximum 8GB in order to work around this when profiling:

node --stack-size=8024 $(which 0x) my-app.js

There may still be a problem opening the flamegraph in Chrome. The same work around can be used by opening Chrome from the command line (platform dependent) and nesting the --stack-size flag within the --js-flags flag: --js-flags="--stack-size 8024".

Debugging

DEBUG=0x* 0x my-app.js

Alternatives

Acknowledgements

Sponsored by nearForm

This tool is inspired from various info and code sources and would have taken much longer without the following people and their Open Source/Info Sharing efforts:

License

MIT

changelog

v4.0.0

  • internal refactor
  • v8 prof is now the default method of stack collection
  • profile folders renamed to {outputDir}/{name}.0x
  • CLI: add --kernel-tracing option
  • CLI: removed --svg flag
  • CLI: removed --gen flag
  • CLI: removed --timestamp-profiles flag
  • CLI: removed --theme flag
  • CLI: removed --include flag
  • CLI: removed --exclude | -x flag
  • CLI: removed --tiers | -t flag
  • CLI: removed --langs | -l flag
  • CLI: renamed --trace-info to --kernel-tracing-debug flag
  • CLI: removed --logging-output flag
  • CLI: removed --stacks-only flag
  • CLI: removed -d | --delay flag
  • CLI: renamed --json-stacks to --tree-debug flag
  • UI: removed langs button
  • UI: removed theme button
  • UI: style changes, minor redesign
  • API: altered mapFrames API (frames is now an array of objects, not strings)
  • UI: rename/reorganize type labels
  • UI: tier coloring improvements
  • UI: search improvements
  • UI: categorization improvements
  • API: removed log option
  • API: added kernelTracing option
  • API: removed svg option
  • API: removed gen option
  • API: removed timestamp-profiles option
  • API: removed theme option
  • API: removed include option
  • API: removed exclude option
  • API: removed tiers option
  • API: removed langs option
  • API: renamed traceInfo to kernelTracingDebug option
  • API: removed loggingOutput option
  • API: removed stacksOnly option
  • API: removed delay option
  • API: renamed jsonStacks to treeDebug option
  • CLI: enhanced status console output (can be overridden in API)
  • UI: added merging capability (v8 prof only)
  • UI: fully responsive
  • added capturing inline data along side v8 prof ("inlinable" functions)
  • profiling REPL no longer supported
  • support for eval'd code - appears under native frames
  • DOCS: readme changes to reflect API changes
  • DOCS: additional /docs folder for in depth topics
  • API: removed quiet (still part of CLI)
  • API: removed silent (still part of CLI)
  • API: removed open (still part of CLI)
  • CLI: add --on-port flag
  • API: add onPort option
  • CLI: removed --phase flag
  • API: removed phase option

v3.4.1

  • break out the ui and stack converter portions into separate modules: d3-fg and stacks-to-json-tree.

v3.4.0

  • introduce experimental --prof-only flag, generates flamegraph based on internal v8 profiling data, without performing kernel tracing at the same time (as --prof-viz does)

v3.3.0

  • introduce experimental --prof-viz flag, generates additional flamegraph based on internal v8 profiling data
  • linux fixes & tidy up

v3.2.0

  • introduce --phase option
  • change --delay from 300 to 0 - not a breaking change because --phase provides the same result (stripping module loading stacks) far more reliably.

v3.1.0

  • enhance --output-dir option with interpolation feature
  • add --output-html option

v3.0.2

  • missing dep

v3.0.1

  • fix auto open functionality for linux (doesn't close browser on process exit)

v3.0.0

  • node 4 support dropped, node 6+ supported
  • changelog.md added
  • refactor
  • profile folder schema change profile-{pid} -> {pid}.flamegraph
  • --timestamp-profiles prefix instead of suffix profile-{pid}-{timestamp} -> {timestamp}-{pid}.flamegraph
  • programmatic API require('0x') and require('0x/cmd')
  • removed --preview option
  • remove --command option and changed -c gen to --gen
  • added --json-stacks option and no longer storing intermediate JSON by default
  • added alias to -q: --quiet
  • added --silent flag
  • added --name flag
  • added --log-output flag
  • added double dash syntax for nesting flags, e.g.: 0x -- node --zero-fill-buffers script.js
  • --node flag removed – now possible with new CLI syntax (0x [flags] -- node [nodeFlags] script.js [scriptFlags])
  • --node-options removed – now possible with new CLI syntax (0x [flags] -- node [nodeFlags] script.js [scriptFlags])
  • CLI argument schema: unrecognized flags are now rejected
  • 0x === 0x -h (use 0x -- node or 0x -- to profile repl)
  • --stacks-only removed, (use --collect-only)
  • add --collect-only and --visualize-only