nodejs / node

Node.js JavaScript runtime ✨🐢🚀✨
https://nodejs.org
Other
107.66k stars 29.62k forks source link

Debug worker_threads #26609

Open cevek opened 5 years ago

cevek commented 5 years ago

There is no possible to debug worker threads, only through event messages to main thread You can't step into this worker in the chrome dev tools

const worker = new Worker('./worker.js', {});

debugger statements also don't work inside thread

addaleax commented 5 years ago

Fwiw, I recently wrote @eugeneo an email about this. We implemented something similar to the Target domain that Chrome uses, but under a different name. I don’t know if the plan is for Devtools to use the Node.js-specific domain, or for us to eventually use the Target domain as well.

/cc @nodejs/v8-inspector

ofrobots commented 5 years ago

/cc @pavelfeldman

pavelfeldman commented 5 years ago

@aslushnikov , @a1ph, do you have cycles to look at it?

pavelfeldman commented 5 years ago

@ak239 , anyone? :) Now that we have the flattened session support settled in chrome land, we can probably name this domain Target and introduce the flattened sessionId dance in Node.

gauravmahto commented 5 years ago

NDB has added support for worker_threads. Do we expect something similar in future? https://github.com/GoogleChromeLabs/ndb/pull/281

addaleax commented 5 years ago

@gauravmahto Unfortunately, DevTools does not support this yet, https://bugs.chromium.org/p/chromium/issues/detail?id=968472 is their open issue for this.

alexkozy commented 5 years ago

There are two separate things:

addaleax commented 5 years ago

@ak239 What do we need to do for the second part?

gauravmahto commented 5 years ago

@ak239 Maybe this helps solve the second part?

alexkozy commented 5 years ago

@addaleax there are three methods available on V8Inspector: storeCurrentStackTrace,externalAsyncTaskStarted,externalAsyncTaskFinished.

To support step into between threads we need to call storeCurrentStackTrace before postMessage, pass V8StackTraceId to another thread and call externalAsyncTaskStarted / finished on another thread when we are processing this message. V8StackTraceId is actually pair of ints. As soon as it is done, when in ndb you press step into on post message, debugger will automatically go to the worker and pause there when worker is about to process message.

Second idea that is implemented in Chrome, it is doing the same for Worker constructor: store at constructor call, and call started/finished for main worker script. When this one is done, user can click step into worker constructor and go to worker script.

connor4312 commented 3 years ago

In case any debugger implementors come across there -- Node has an extension to the debugger protocol which allows debugging worker threads -- with minimal work given your debugger is already multi-target aware, like VS Code's. However this is not documented anywhere so I have no idea how stable it may be, YMMV.

AriPerkkio commented 1 year ago

Debugging workers with --inspect-brk is possible with some hacky patching. Passing it as execArgv won't work but using node:inspector inside the worker works.

This does not work:

import { isMainThread, Worker } from "node:worker_threads";
import { fileURLToPath } from "node:url";

const filename = fileURLToPath(import.meta.url);

if (isMainThread) {
  new Worker(filename, { execArgv: ["--inspect-brk"] });
} else {
    debugger; // No stop here
}

The node:inspector works. Chrome DevTools stop at the debugger.

import { isMainThread, Worker } from "node:worker_threads";
import { fileURLToPath } from "node:url";
+import inspector from "node:inspector";

const filename = fileURLToPath(import.meta.url);

if (isMainThread) {
  new Worker(filename, { execArgv: ["--inspect-brk"] });
} else {
+  if (process.execArgv.includes("--inspect-brk")) {
+    inspector.open();
+    inspector.waitForDebugger();
+  }

  debugger; // Stops in worker thread!
}

Adding the process.execArgv checks in userland seems like unnecessary work - maybe Node could do this for us?

danieltroger commented 1 year ago

Debugging workers with --inspect-brk is possible with some hacky patching. Passing it as execArgv won't work but using node:inspector inside the worker works.

Could you please elaborate on this? Does this enable debugging a worker with chrome devtools? Because I'm not having any success - the worker doesn't show up in chrome after executing inspector.open();inspector.waitForDebugger(); in it, it seems like it just freezes

AriPerkkio commented 1 year ago

Does this enable debugging a worker with chrome devtools?

Yes, just save the code above into file and run with node. Here's picture from Chrome dev tools:

image

danieltroger commented 1 year ago

Hmm, @AriPerkkio do you know if it works with require too? Just adding const inspector = require("node:inspector"); at the top of the worker I'm trying to debug, https://github.com/nodejs/node/blob/55d2eb53d78040472371eedd0e9e7922748dfb0e/lib/internal/modules/esm/worker.js makes the worker freeze for me (become unresponsive at 0% CPU usage) 😢

And nothing shows up in chrome

Screenshot 2023-11-07 at 19 47 19

I'm trying to get a debugger to where this error is thrown, see reproduction steps there: https://github.com/swc-project/swc-node/issues/736#issue-1962957988 and I'm following these steps to build node 20.9.0 https://www.devdungeon.com/content/build-nodejs-source

connor4312 commented 1 year ago

Fyi, VS Code's JavaScript debugger automatically debugs node_workers using the NodeWorker domain (https://github.com/nodejs/node/issues/26609#issuecomment-742180739). So if you just need to debug a worker, easiest way is to run "Create JavaScript Debug Terminal" in VS Code; everything run in that terminal will be debugged.

danieltroger commented 1 year ago

@connor4312 TYSM, that got me a debugger inside of the worker!

Unfortunately it doesn't seem like the vscode debugger can go to function definitions on all functions ( [[FunctionLocation]]) and also the "break on caught exceptions" doesn't seem to work within a worker but I got further, thanks :)

thoughtsunificator commented 1 week ago

But what about the CLI debugger?

I'd like to be able to pass inspect as a positional argument to the worker causing it to start the CLI debugger just like we would do with node inspect script.js.

As things are, I am unable to start the internal (CLI) debugger for a worker thread on Node v22.9.0 (Linux) with :

node inspect-cli.js

const { Worker, isMainThread } = require('node:worker_threads');
if(isMainThread) {
    const worker = new Worker(__filename, { execArgv: ["--inspect-brk"]   })
    worker.on('error', console.error)
    worker.on('exit', (code) => console.log(`Worker stopped with exit code ${code}`))
} else {
    debugger
}

What happens is, the process hangs, and here are the NODE_DEBUG and strace logs:

node_debug.txt strace.txt

Here's what the debugger yields when running the scripts command after attaching a debugger client using node inspect -p PID:

Spoiler
[user@pc node-playground]$ node inspect -p 123
connecting to 127.0.0.1:9229 ... ok
debug> scripts
  22: node:events
  24: node:buffer
  31: node:async_hooks
  32: node:timers
  34: node:path
  37: node:querystring
  42: node:fs
  47: node:util
  62: node:url
  66: node:diagnostics_channel
  75: inspect-cli.js
  76: node:worker_threads
  80: node:stream
  92: node:string_decoder
  95: node:stream/promises
  101: node:inspector
  102: node:tty
  103: node:net

Finally, I think it would only make sense that nodejs start the CLI debugger: