It's a common pattern to implement timeout of some asynchronous function, using deffered/promise:
// Create a Deferred and return its Promise
function timeout(funct, args, time) {
var dfd = new jQuery.Deferred();
// execute asynchronous code
funct.apply(null, args);
// When the asynchronous code is completed, resolve the Deferred:
dfd.resolve('success');
setTimeout(function() {
dfd.reject('sorry');
}, time);
return dfd.promise();
}
Now we can execute some asynchronous function called myFunc
and handle timeout:
// Attach a done and fail handler for the asyncEvent
$.when( timeout(myFunc, [some_args], 1000) ).then(
function(status) {
alert( status + ', things are going well' );
},
function(status) {
alert( status + ', you fail this time' );
}
);
OK, let's make a twist in this story! Imagine that the myFunc
itself returns a promise (NOTE: promise NOT deferred and I can't change it):
function myFunc(){
var dfd = new jQuery.Deffered();
superImportantLibrary.doSomething(function(data)){
if(data.length < 5){
dfd.reject('too few data');
}
else{
dfd.resolve('success!');
}
}, {'error_callback': function(){
dfd.reject("there was something wrong but it wasn't timeout");}
}});
return dfd.promise();
}
Now if I wrap myFunc
in timeout
, I will loose the ability to handle errors different then timeout. If myFunc
emit progress events, I will loose this as well.
So the question is: how to modify timeout
function so it can accept functions returning promises without loosing their errors/progress information?
I realize this is 2 years old, but in case someone is looking for the answer...
I think Benjamin was close in that you'll want your timeout to be handled separately, so we'll start with his delay function.
Then, if you wanted to wait before code is executed you can call the method you want delayed as a result of this promise.
This is usually what I'm trying to do when I go looking for a refresher (why I'm here). However, the question was not about delaying the execution, but throwing an error if it took too long. In that case, this complicates things because you don't want to wait around for the timeout if you don't have to, so you can't just wrap the two promises in a "when". Looks like we need another deferred in the mix. (See Wait for the first of multiple jQuery Deferreds to be resolved?)
We can streamline this, knowing that in this case the master defer only rejects if the timeout happens first and only resolves if the functPromise resolves first. Because of this, we don't need to pass the functPromise to the master defer resolve, because it's the only thing that could be passed and we're still in scope.
You should always promsiify at the lowest level possible. Let's start from the basics.
I'll use jQuery promises here, but this should really be done with a stronger library like Bluebird Let's start simple, by creating our
delay
as:Note delay doesn't do anything surprising, all our delay function does is cause a delay of
ms
milliseconds.Now, for your library, we want to create a version of
doSomething
that works with promises:Note both our delay and
doSomethingAsync
functions both do just one thing. Now the fun begins.Now in Bluebird, this whole code would have been: