Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

[worker_threads]: Main thread receives Error object when the worker throws a primitive value #35506

Open
suin opened this issue Oct 5, 2020 · 12 comments
Labels
confirmed-bug Issues with confirmed bugs. eventtarget Issues and PRs related to the EventTarget implementation. worker Issues and PRs related to Worker support.

Comments

@suin
Copy link

suin commented Oct 5, 2020

  • Version: v14.7.0
  • Platform: Darwin suin 18.7.0 Darwin Kernel Version 18.7.0: Mon Feb 10 21:08:45 PST 2020; root:xnu-4903.278.28~1/RELEASE_X86_64 x86_64
  • Subsystem: N/A

What steps will reproduce the bug?

Summary: Since Node v14.7.0, when primitive values are thrown in a worker, the main thread receives it as Error objects.

I'm not sure this is actually a bug. But I couldn't find out the information about this breaking change. So I have created this issue. If this is an expected change, please close this issue.

// main.js
const { Worker } = require('worker_threads')
const worker = new Worker('./worker.js')
console.log(process.versions.node)
worker.on('error', err => {
  console.error(err)
})
// worker.js
throw 'Error was thrown in worker'

in Node v14.6.0:

$ node main.js
14.6.0
Error was thrown in worker

in Node v14.7.0:

$ node main.js
14.7.0
Error [ERR_UNHANDLED_ERROR]: Unhandled error. ('Error was thrown in worker')
    at process.emit (events.js:303:17)
    at emitUnhandledRejectionOrErr (internal/event_target.js:541:11)
    at MessagePort.[nodejs.internal.kHybridDispatch] (internal/event_target.js:356:9)
    at MessagePort.exports.emitMessage (internal/per_context/messageport.js:18:26) {
  code: 'ERR_UNHANDLED_ERROR',
  context: 'Error was thrown in worker'
}

Other primitives:

Undefined

// worker.js
throw undefined
$ node main.js
14.6.0
undefined

$ node main.js
14.7.0
Error [ERR_UNHANDLED_ERROR]: Unhandled error. (undefined)
    at process.emit (events.js:303:17)
    at emitUnhandledRejectionOrErr (internal/event_target.js:541:11)
    at MessagePort.[nodejs.internal.kHybridDispatch] (internal/event_target.js:356:9)
    at MessagePort.exports.emitMessage (internal/per_context/messageport.js:18:26) {
  code: 'ERR_UNHANDLED_ERROR',
  context: undefined
}

Number

// worker.js
throw 0
$ node main.js
14.6.0
0

$ node main.js
14.7.0
Error [ERR_UNHANDLED_ERROR]: Unhandled error. (0)
    at process.emit (events.js:303:17)
    at emitUnhandledRejectionOrErr (internal/event_target.js:541:11)
    at MessagePort.[nodejs.internal.kHybridDispatch] (internal/event_target.js:356:9)
    at MessagePort.exports.emitMessage (internal/per_context/messageport.js:18:26) {
  code: 'ERR_UNHANDLED_ERROR',
  context: 0
}

Symbol

// worker.js
throw Symbol('this is a symbol')
$ node main.js
14.6.0
Symbol(this is a symbol)

$ node main.js
14.7.0
Error [ERR_UNHANDLED_ERROR]: Unhandled error. (Symbol(this is a symbol))
    at process.emit (events.js:303:17)
    at emitUnhandledRejectionOrErr (internal/event_target.js:541:11)
    at MessagePort.[nodejs.internal.kHybridDispatch] (internal/event_target.js:356:9)
    at MessagePort.exports.emitMessage (internal/per_context/messageport.js:18:26)

How often does it reproduce? Is there a required condition?

There is no condition.

What is the expected behavior?

I'm not sure which is valid behavior, but at point of view of backward compatibilities, it would be the expected behavior that the main thread receives the unhandled primitive error as the primitive value instead of the Error object like the Node v14.6.0 behavior.

What do you see instead?

The main thread gets the Error object.

Additional information

@aduh95
Copy link
Contributor

aduh95 commented Oct 5, 2020

Interestingly, when using data: URL, the primitive object is passed to the error handler as in 14.6.0:

const { Worker } = require('worker_threads');

console.log(process.versions.node);
new Worker(new URL('data:text/javascript,throw 4')).on('error', err => {
  console.error(err);
});
$ node main.js
15.0.0-pre
4

@codebytere
Copy link
Member

codebytere commented Oct 8, 2020

Looks like a consequence of 0aa3809b6b - i don't think this should be expected behavior but @addaleax can maybe confirm?

@addaleax addaleax added confirmed-bug Issues with confirmed bugs. worker Issues and PRs related to Worker support. eventtarget Issues and PRs related to the EventTarget implementation. labels Oct 8, 2020
@addaleax
Copy link
Member

addaleax commented Oct 8, 2020

Right – this should not be happening, it’s a bug 👍

@addaleax
Copy link
Member

Ok, so basically the problem here is that, because the main script inside a Worker is started from within a .on('message') event handler, it ends up in the EventTarget error handling chain, which is basically emitting a process.on('error') event first, and if there are no handlers, wrapping the exception and treating it as an uncaught one.

I basically see 2 ways to solve this:

  1. Delay the execution of the LOAD_SCRIPT handler with a nextTick. This would make errors thrown inside of it it, including during the main script, be emitted as regular uncaught exceptions.
  2. Don’t use the emitUnhandledRejectionOrErr() mechanism at all for Node.js-style event listeners. It’s somewhat unexpected that the exception handling behavior of a object.on('foo') listener depends on whether object is an EventEmitter or NodeEventTarget.

In particular, this did not just change the behavior for the main script unintentionally, but also for exceptions thrown inside any port.on('message') listeners.

@jasnell @benjamingr Thoughts?

@addaleax
Copy link
Member

ping @jasnell @benjamingr any further thoughts? I’d lean towards option 2, if nobody has a strong opinion.

@benjamingr
Copy link
Member

I lean towards option #2 as well with no strong opinions.

@jasnell
Copy link
Member

jasnell commented Jan 25, 2021

Missed the original ping on this... no strong opinions on either option.

@yashLadha
Copy link
Contributor

This is not coming in current master, should we go ahead and create a minor version patch for this?

@benjamingr
Copy link
Member

@yashLadha do you mind clarifying what you mean by that?

@yashLadha
Copy link
Contributor

@benjamingr I meant to say that if this issue exists in an older version of node we can backport the fix to that tree, if it feels viable.

@benjamingr
Copy link
Member

@yashLadha was this issue actually resolved in master? I don't see any PRs or work relating to it.

@kaydenvanrijn
Copy link

@yashLadha was this issue actually resolved in master? I don't see any PRs or work relating to it.

Any update on this that you know about? I've encountered an issue when handling errors thrown within a worker thread in v20

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
confirmed-bug Issues with confirmed bugs. eventtarget Issues and PRs related to the EventTarget implementation. worker Issues and PRs related to Worker support.
Projects
None yet
Development

No branches or pull requests

8 participants