Détail du package

dtrace-provider

chrisa8.2mBSD-2-Clause0.8.8

Native DTrace providers for node.js applications

dtrace, usdt

readme

dtrace-provider - Native DTrace providers for Node.js apps.

This extension allows you to create native DTrace providers for your Node.js applications. That is, to create providers and probes which expose information specific to your application, rather than information about the node runtime.

You could use this to expose high-level information about the inner workings of your application, or to create a specific context in which to look at information from other runtime or system-level providers.

The provider is not created in the usual way, by declaring it and then changing the build process to include it, but instead dynamically at runtime. This is done entirely in-process, and there is no background compiler or dtrace(1M) invocation. The process creating the provider need not run as root.

INSTALL

$ npm install dtrace-provider

EXAMPLE

Here's a simple example of creating a provider:

var d = require('dtrace-provider');

var dtp = d.createDTraceProvider("nodeapp");
var p1 = dtp.addProbe("probe1", "int", "int");
var p2 = dtp.addProbe("probe2", "char *");
dtp.enable();

Probes may be fired via the provider object:

dtp.fire("probe1", function() {
    return [1, 2];
});
dtp.fire("probe2", function() {
    return ["hello, dtrace via provider", "foo"];
});

or via the probe objects themselves:

p1.fire(function() {
  return [1, 2, 3, 4, 5, 6];
});
p2.fire(function() {
  return ["hello, dtrace via probe", "foo"];
});

Note that .fire() takes a callback that returns the arguments to be provided when the DTrace probe actually fires. This allows you to call .fire() unconditionally when you want to fire the probe, but the callback will be invoked only when the DTrace probe is actually enabled. This allows you to create probes whose arguments might be expensive to construct, and only do any work when the probe is actually enabled. (Examples might include converting a large object to a string representation or gathering large amounts of information.)

In some cases, creating a new closure to pass to .fire() each time it's called may introduce unwanted overhead. For extremely CPU-intensive or memory-conscious workloads, you can avoid this by lifting the closures for your hot probes into an outer scope. You can then supply arguments to that function as additional arguments to .fire(). As an example, you can convert the following program:

function manipulateObj(largeObj) {
    var count = 0;
    var name = null;
    ...
    p1.fire(function () {
        return [count, keyToValue(name), JSON.stringify(largeObj)];
    });
}

Into this one:

function f(a, b, c) {
    return [a, keyToValue(b), JSON.stringify(c)];
}

function manipulateObj(largeObj) {
    var count = 0;
    var name = null;
    ...
    p1.fire(f, count, name, largeObj);
}

Be careful to avoid passing .fire() additional arguments that are themselves expensive to construct, as that undermines the design goal here: minimizing the effect of disabled probes.

This example creates a provider called "nodeapp", and adds two probes. It then enables the provider, at which point the provider becomes visible to DTrace.

The probes are then fired, which produces this output:

$ sudo dtrace -Z -n 'nodeapp*:::probe1{ trace(arg0); trace(arg1) }'  \
                 -n 'nodeapp*:::probe2{ trace(copyinstr(arg0));  }'
dtrace: description 'nodeapp*:::probe1' matched 0 probes
dtrace: description 'nodeapp*:::probe2' matched 0 probes
CPU     ID                    FUNCTION:NAME
  1 123562                      func:probe1                 1                2
  1 123563                      func:probe2   hello, dtrace                    

Arguments are captured by a callback only executed when the probe is enabled. This means you can do more expensive work to gather arguments.

The maximum number of arguments supported is 32.

Available argument types are "int", for integer numeric values, "char *" for strings, and "json" for objects rendered into JSON strings.

Arguments typed as "json" will be created as "char *" probes in DTrace, but objects passed to these probe arguments will be automatically serialized to JSON before being passed to DTrace. This feature is best used in conjunction with the json() D subroutine, but is available whether or not the platform supports it.

# create a json probe:

var dtp = d.createDTraceProvider("nodeapp");
var p1 = dtp.addProbe("j1", "json");
dtp.enable();
p1.fire(function() { return { "foo": "bar" }; });

# on a platform supporting json():

$ sudo dtrace -Z -n 'nodeapp*:::j1{ this->j = copyinstr(arg0); \
                                    trace(json(this->j, "foo")) }'
dtrace: description 'nodeapp$target:::j1' matched 0 probes
CPU     ID                    FUNCTION:NAME
  0  68712                            j1:j1   bar

PLATFORM SUPPORT

This libusdt-based Node.JS module supports 64 and 32 bit processes on Mac OS X and Solaris-like systems such as illumos or SmartOS. As more platform support is added to libusdt, those platforms will be supported by this module. See libusdt's status at:

https://github.com/chrisa/libusdt#readme

When using Mac OS X, be aware that as of 10.11 (El Capitan), DTrace use is restricted, and you'll probably want to disable SIP to effectively use DTrace.

FreeBSD 10 and 11 are also supported, but you'll need to make sure that you have the DTrace headers installed in /usr/src otherwise libusdt won't be able to compile. You can clone them using SVN, or find the correct src.txz here and extract that. Also note that FreeBSD 10 is restricted to only 4 working arguments per probe.

Platforms not supporting DTrace (notably, Linux and Windows) may install this module without building libusdt, with a stub no-op implementation provided for compatibility. This allows cross-platform npm modules to embed probes and include a dependency on this module.

GNU Make is required to build libusdt; the build scripts will look for gmake in PATH first, and then for make.

TROUBLESHOOTING BUILD ISSUES

If compilation fails during installation on platforms with DTrace, then the library will fall back to the stub implementation that does nothing. To force an installation failure when compiling fails, set the environment variable NODE_DTRACE_PROVIDER_REQUIRE to hard:

$ NODE_DTRACE_PROVIDER_REQUIRE=hard npm install

This will then show you the output of the build process so you can see at which point it's having an issue. Common issues are:

  • Missing a C/C++ compiler toolchain for your platform.
  • python is Python 3 instead of Python 2; run npm config set python python2.7 (or similar) to set the Python binary npm uses.
  • On OS X you may need to agree to the XCode license if that's the compiler toolchain you're using. This will usually manifest with an error like Agreeing to the Xcode/iOS license requires admin privileges, please re-run as root via sudo. To accept the license, you can run sudo xcodebuild -license.

Once you've found and fixed the issue, you can run npm rebuild to rerun the lifecycle scripts.

CAVEATS

There is some overhead to probes, even when disabled. Probes are already using the "is-enabled" feature of DTrace to control execution of the arguments-gathering callback, but some work still needs to be done before that's checked. This overhead should not be a problem unless probes are placed in particularly hot code paths.

CONTRIBUTING

To clone the project's source code:

$ git clone --recursive https://github.com/chrisa/node-dtrace-provider.git

For issues, please use the GitHub issue tracker linked to the repository. GitHub pull requests are very welcome.

RUNNING THE TESTS

$ npm install
$ sudo ./node_modules/.bin/tap --tap test/*.test.js

OTHER IMPLEMENTATIONS

This node extension is derived from the ruby-dtrace gem, via the Perl module Devel::DTrace::Provider, both of which provide the same functionality to those languages.

changelog

dtrace-provider - Changes

HISTORY

  • 0.8.8: Known support for v0.10.48, v0.12.16, v4.8.1, v6.17.0, v7.5.0, v8.16.0, v9.3.0, v10.16.0, v12.7.0 (#125)

  • 0.8.7: Known support for v0.10.48, v0.12.16, v4.6.0, v7.5.0, v8.9.4, v10.3.0 (#119) Don't crash when attempting to fire unknown probes (#120)

  • 0.8.6: Improved compilation failure behaviour (#96)

  • 0.8.5: Reverted "Install fails on Debian due to differently named node binary" for now

  • 0.8.4: Only log error once when DTraceProviderBindings can't be found Install fails on Debian due to differently named node binary

  • 0.8.3: Install fails with yarn

  • 0.8.2: Error installing in 64-bit SmartOS zones with 32-bit node

  • 0.8.1: Support FreeBSD 10 & 11

  • 0.8.0: Support passing additional arguments to probe function via .fire()

  • 0.7.1: Update libusdt for chrisa/libusdt#12 fix

  • 0.7.0: known support for v0.10.47, v0.12.16, v4.6.0. Updated NaN dependency to remove warnings on newer Node versions.

  • 0.2.8: Add NODE_MODULE() declaration for compatibility with Node 0.9.1+ (reported by Trent Mick) Remove execSync dependency from tests.

  • 0.2.7: Don't build on FreeBSD by default - DTrace is not yet built in releases.

  • 0.2.6: Fall back to make(1) if gmake(1) is unavailable, still expected to be GNU Make (Trent Mick)

  • 0.2.5: Add "json" probe argument type, automatically serialising objects as JSON Trust npm to set PATH appropriately when invoking node (reported by Dave Pacheco) libusdt update - allow provider memory to be freed (reported by Bryan Cantrill) Build libusdt with gmake by default (reported by Keith Wesolowski) Turn the various scripts in test/ into a TAP-based testsuite.

  • 0.2.4: Improve Node architecture detection to support 0.6.x, and respect npm's prefix when choosing a node binary to use (reported by Trent Mick)

  • 0.2.3: libusdt update - don't invoke ranlib on SunOS-derived systems Disambiguate module name in probe tuple, and optionally allow it to be specified when creating a provider. (Bryan Cantrill bcantrill@acm.org)

  • 0.2.2: libusdt update for build fixes Respect MAKE variable in build script

  • 0.2.1: Update binding.gyp for clang on Snow Leopard - no space after -L.

  • 0.2.0: Update libusdt, and attempt to build it correctly for various platforms. Add support for disabling providers and removing probes.

  • 0.1.1: Replace Node-specific implementation with wrappers for libusdt. Extend argument support to 32 primitives. Adds Solaris x86_64 support.

  • 0.0.9: Force the build architecture to x86_64 for OS X.

  • 0.0.8: Removed overridden "scripts" section from package.json, breaking Windows installs

  • 0.0.7: Fix for multiple enable() calls breaking providers.

  • 0.0.6: Fix for segfault trying to use non-enabled probes (Mark Cavage mcavage@gmail.com)

  • 0.0.5: Revert changes to make probe objects available.

  • 0.0.4: Remove unused "sys" import (Alex Whitman) No longer builds an empty extension on non-DTrace platforms Probe objects are made available to Javascript.

  • 0.0.3: Builds to a stubbed-out version on non-DTrace platforms (Mark Cavage mcavage@gmail.com)

  • 0.0.2: Solaris i386 support. Fixes memory leaks Improved performance, enabled- and disabled-probe.

  • 0.0.1: First working version: OSX x86_64 only.