Their is already a question on this topic
Node.js Best Practice Exception Handling
which is old and answers are very much outdated, domains
have even deprecated since then.
Now in a post Async/Await Node.js scenario shouldn't we consider sync and async cases similarly and throw exceptions in sync functions and rejecting promises in async functions instead of returning an Error
instance in the former case.
let divideSync = function(x,y) {
// if error condition?
if ( y === 0 ) {
// "throw" the error
throw new Error("Can't divide by zero exception")
}
else {
// no error occured, continue on
return x/y
}
}
Simulating async divide operation
let divideAsync = function(x, y) {
return new Promise(function(resolve, reject) {
setTimeout(function() {
// if error condition?
if (y === 0) {
// "throw" the error safely by rejecting the promise
reject (new Error("Can't divide by zero exception"));
} else {
// no error occured, continue on
resolve(x / y)
}
}, 1000);
})
};
So sync and async exceptions can be handled in a uniform manner
let main = async function () {
try {
//const resultSync = divideSync(4,0);
const resultAsync = await divideAsync(4,0);
}
catch(ex) {
console.log(ex.message);
}
}
The answers from Node.js Best Practice Exception Handling are old and very much outdated
Not that much. This answer, with the list from this well-maintained blog post, is quite up to date.
The offical node.js guide is always a good read, and the general approach hasn't changed that much.
So what has changed?
async
/await
. (Note: event emitters etc are a different case)process.on('uncaughtException')
is supplemented by process.on('unhandledRejection')
So what does this mean for the common code?
Shouldn't we consider sync and async cases similarly and throw exceptions in sync functions and rejecting promises in async functions instead of returning an
Error
instance?
Yes, exactly. You should reject your promises with Error
s (or throw
them from async function
s).
Notice you will rarely have to call reject
yourself. With promises, you should be able to throw
in your code. If you can't, you're likely not using them correctly - and programmer mistakes wouldn't be caught either there.
The golden rule for this code is: Never use callbacks that are not promise callbacks. "Promise callbacks" refers to the new Promise
, then
, and catch
arguments, and possibly some of the custom methods of your library (e.g. finally
). This is where your example code above has a problem. Written correctly, it should read
async function divideAsync(x, y) {
await new Promise(resolve =>
setTimeout(resolve, 1000) // don't even pass a function expression
);
if (y === 0) {
throw new Error("Can't divide by zero exception");
} else {
return x / y;
}
}
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With