Package Exports
- @sentry-internal/node-native-stacktrace
- @sentry-internal/node-native-stacktrace/lib/index.js
This package does not declare an exports field, so the exports above have been automatically detected and optimized by JSPM instead. If any package subpath is missing, it is recommended to post an issue to the original package (@sentry-internal/node-native-stacktrace) to support the "exports" field. If that is not possible, create a JSPM override to customize the exports field for this package.
Readme
@sentry-internal/node-native-stacktrace
A native Node.js module that can capture JavaScript stack traces for registered main or worker threads from any other thread, even if event loops are blocked.
The module also provides a means to create a watchdog system to track event loop blocking via periodic heartbeats. When the time from the last heartbeat crosses a threshold, JavaScript stack traces can be captured. The heartbeats can optionally include state information which is included with the corresponding stack trace.
This native module is used for Sentry's Event Loop Blocked Detection feature.
Basic Usage
1. Register threads you want to monitor
In your main thread or worker threads:
import { registerThread } from "@sentry-internal/node-native-stacktrace";
// Register this thread for monitoring
registerThread();
2. Capture stack traces from any thread
import { captureStackTrace } from "@sentry-internal/node-native-stacktrace";
// Capture stack traces from all registered threads
const stacks = captureStackTrace();
console.log(stacks);
Example Output
Stack traces show where each thread is currently executing:
{
'0': { // Main thread has ID '0'
frames: [
{
function: 'from',
filename: 'node:buffer',
lineno: 298,
colno: 28
},
{
function: 'pbkdf2Sync',
filename: 'node:internal/crypto/pbkdf2',
lineno: 78,
colno: 17
},
{
function: 'longWork',
filename: '/app/test.js',
lineno: 20,
colno: 29
},
{
function: '?',
filename: '/app/test.js',
lineno: 24,
colno: 1
}
]
},
'2': { // Worker thread
frames: [
{
function: 'from',
filename: 'node:buffer',
lineno: 298,
colno: 28
},
{
function: 'pbkdf2Sync',
filename: 'node:internal/crypto/pbkdf2',
lineno: 78,
colno: 17
},
{
function: 'longWork',
filename: '/app/worker.js',
lineno: 10,
colno: 29
},
{
function: '?',
filename: '/app/worker.js',
lineno: 14,
colno: 1
}
]
}
}
Advanced Usage: Automatic blocked event loop Detection
Set up automatic detection of blocked event loops:
1. Set up thread heartbeats
Send regular heartbeats with optional state information:
import {
registerThread,
threadPoll,
} from "@sentry-internal/node-native-stacktrace";
// Register this thread
registerThread();
// Send heartbeats every 200ms with optional state
setInterval(() => {
threadPoll({
endpoint: "/api/current-request",
userId: getCurrentUserId(),
});
}, 200);
2. Monitor from a watchdog thread
Monitor all registered threads from a dedicated thread:
import {
captureStackTrace,
getThreadsLastSeen,
} from "@sentry-internal/node-native-stacktrace";
const THRESHOLD = 1000; // 1 second
setInterval(() => {
const threadsLastSeen = getThreadsLastSeen();
for (const [threadId, timeSinceLastSeen] of Object.entries(threadsLastSeen)) {
if (timeSinceLastSeen > THRESHOLD) {
// Thread appears to be blocked - capture diagnostics
const stackTraces = captureStackTrace();
const blockedThread = stackTraces[threadId];
console.error(`🚨 Thread ${threadId} blocked for ${timeSinceLastSeen}ms`);
console.error("Stack trace:", blockedThread.frames);
console.error("Last known state:", blockedThread.state);
}
}
}, 500); // Check every 500ms
API Reference
Functions
registerThread(threadName?: string): void
Registers the current thread for monitoring. Must be called from each thread you want to capture stack traces from.
threadName
(optional): Name for the thread. Defaults to the current thread ID.
captureStackTrace<State>(): Record<string, Thread<State>>
Captures stack traces from all registered threads. Can be called from any thread but will not capture the stack trace of the calling thread itself.
type Thread<S> = {
frames: StackFrame[];
state?: S;
};
type StackFrame = {
function: string;
filename: string;
lineno: number;
colno: number;
};
threadPoll<State>(state?: State, disableLastSeen?: boolean): void
Sends a heartbeat from the current thread with optional state information. The state object will be serialized and included as a JavaScript object with the corresponding stack trace.
state
(optional): An object containing state information to include with the stack trace.disableLastSeen
(optional): Iftrue
, disables the tracking of the last seen time for this thread.
getThreadsLastSeen(): Record<string, number>
Returns the time in milliseconds since each registered thread called
threadPoll()
.