There is an interesting issue with handling Exception with Promises, using setTimeout
, that I can't wrap my head around.
Example:
function boo() {
return new Promise(function (resolve, reject) {
setTimeout(function () {
throw new Error("Error !!!!!!!!!!!!!!!!!!!!!!!!!!!;");
resolve("You're my hero !");
}, 2 * 1000);
});
};
The code above will throw an error after 2 seconds, so according to the Chapter 3 from async & performance we should be able to catch that error in catch
method or in rejection handler from the then
method of the Promise.
boo()
.then(function (val) {
debugger;
console.log("THEN FULLFILLED");
},
function (rejVal) {
debugger; // <--- Should catch the Error here
console.log("THEN REJECTION");
})
.catch(function (er) {
debugger; // <--- or here
console.log("CATCH");
});
The interesting thing is that exception is thrown in global context and never gets into catch
or rejection handler from then
method.
But, if we remove setTimeout
from the example,
i.e.
function boo() {
return new Promise(function (resolve, reject) {
//setTimeout(function () {
throw new Error("Error !!!!!!!!!!!!!!!!!!!!!!!!!!!;");
//resolve("You're my hero !");
//}, 2 * 1000);
});
};
This will work as expacted and reject handler from the then
method is called.
I have commented the usage of throwing error and called reject with error message.
function boo() {
return new Promise(function (resolve, reject) {
setTimeout(function () {
/*throw new Error("Error !!!!!!!!!!!!!!!!!!!!!!!!!!!;");
resolve("You're my hero !");*/
reject("Error");
}, 2 * 100);
});
};
Please let me know if I have understood your intention correctly.
@deeparaveendran thank you for your time and message. What I was trying to achieve is to simulate an async request, this is not necessarily an Ajax request, just any async operation that could take some time.
I was wondering what would happen if I throw an Error there.
I was sure that the catch
block should, well catch it but it didn't happen. That is why I opened that issue hopping that @getify could help me with this.
Regarding your statement that: "_Using throw directly on the setTimeout will take preference and will throw error globally...._" could you please elaborate more, I couldn't find any information about it.
When you use the setTimeout, the promise constructor has already completed, synchronously, before you throw the exception... so it obviously cannot catch it after the fact. More precisely, exceptions are synchronous to their execution stack, and you've moved the exception to a separate execution stack by wrapping it in the timeout.
@Konrud Kyle Simpson has explained very clearly and that is what I meant too. Thank you @getify
(clearing out issues for second edition work)
Most helpful comment
When you use the setTimeout, the promise constructor has already completed, synchronously, before you throw the exception... so it obviously cannot catch it after the fact. More precisely, exceptions are synchronous to their execution stack, and you've moved the exception to a separate execution stack by wrapping it in the timeout.