How to debug Node.JS child forked process?

21,433

Solution 1

It is a known bug in node.js that has been recently fixed (although not backported to v0.10).

See this issue for more details: https://github.com/joyent/node/issues/5318

There is a workaround where you alter the command-line for each worker process, although the API was not meant to be used this way (the workaround might stop working in the future). Here is the source code from the github issue:

var cluster = require('cluster');
var http = require('http');

if (cluster.isMaster) {
  var debug = process.execArgv.indexOf('--debug') !== -1;
  cluster.setupMaster({
    execArgv: process.execArgv.filter(function(s) { return s !== '--debug' })
  });
  for (var i = 0; i < 2; ++i) {
    if (debug) cluster.settings.execArgv.push('--debug=' + (5859 + i));
    cluster.fork();
    if (debug) cluster.settings.execArgv.pop();
  }
}
else {
  var server = http.createServer(function(req, res) {
    res.end('OK');
  });
  server.listen(8000);
}

Solution 2

Yes. You have to spawn your process in a new port. There is a workaround to debug with clusters, in the same way you can do:

Start your app with the --debug command and then:

var child = require('child_process');
var debug = typeof v8debug === 'object';
if (debug) {   
    //Set an unused port number.    
    process.execArgv.push('--debug=' + (40894));    
}    
child.fork(__dirname + '/task.js');

debugger listening on port 40894

Solution 3

Quick simple fix ( where using chrome://inspect/#devices )

var child = require('child_process');
child.fork(__dirname + '/task.js',[],{execArgv:['--inspect-brk']});

Then run your app without any --inspect-brk and the main process won't debug but the forked process will and no conflicts.

To stop a fork conflicting when debugging the main process ;

child.fork(__dirname + '/task.js',[],{execArgv:['--inspect=xxxx']});

where xxxx is some port not being used for debugging the main process. Though I haven't managed to easily connect to both at the same time in the debugger even though it reports as listening.

Solution 4

I find that setting the 'execArgv' attribute in the fork func will work:

const child = fork('start.js', [], {
cwd: startPath,
silent: true,
execArgv: ['--inspect=10245'] });

Solution 5

if "process.execArgv" doenst work you have to try:

if (debug) {
    process.argv.push('--debug=' + (40894));
}

this worked for me..

Share:
21,433
Guy Korland
Author by

Guy Korland

Updated on November 11, 2020

Comments

  • Guy Korland
    Guy Korland over 3 years

    I'm trying to debug the child Node.JS process created using:

    var child = require('child_process');
    child .fork(__dirname + '/task.js');
    

    The problem is that when running in IntelliJ/WebStorm both parent and child process start on the same port.

    debugger listening on port 40893
    debugger listening on port 40893
    

    So it only debugs the parent process.

    Is there any way to set IntelliJ to debug the child process or force it to start on a different port so I can connect it in Remote debug?

  • Guy Korland
    Guy Korland almost 11 years
    Is there a way to achieve the same thing with child.fork()?
  • Miroslav Bajtoš
    Miroslav Bajtoš almost 11 years
    @GuyKorland Of course, just add --debug={port} to command-line arguments of your child process.
  • lena
    lena almost 11 years
    see also this comment
  • Rémy DAVID
    Rémy DAVID over 9 years
    I struggled on this because I affected the child_process module to a local variable called process, hiding the node process global variable, hence process.execArgv was not defined. Beaware "process" is a node global variable!
  • MorningDew
    MorningDew over 9 years
    I'm not sure why, but my process.execArgv is an array, I had to modify it to process.execArgv.toString().indexOf('--debug') !== -1; in order to get it to work. Thanks!
  • Oleg Belousov
    Oleg Belousov almost 9 years
    Using node 0.12.1, problem still exists, the condition did not return true(checking for debug), but changing the port the way that you suggested worked.
  • Liang Zhou
    Liang Zhou over 7 years
    --inspect is the modern way, but it doesn't has much to do with the problem here - debugging a child process.
  • Breedly
    Breedly over 6 years
    Chrome will not discover the extra ports automatically. You have to tell it to connect with them manually.
  • qbert65536
    qbert65536 over 5 years
    This worked great for me when nothing else did kudos.
  • tnrich
    tnrich over 4 years
    For other's trying to use spawn but not having it work, I find that this works for me: child.spawn('node',['--inspect-brk', __dirname + '/task.js']);