Throwing an Error in jQuery's Deferred object

15,437

Now updated for jQuery 1.8+

The easiest way to tackle this is to run the response of $.ajax through .then to filter based on success or failure of the data.

$.ajax()
    .then(function (response) {
        return $.Deferred(function (deferred) {
            var problem = hasError(response);

            if (problem) {
                return deferred.reject(problem)
            }

            deferred.resolve(response);
        }).promise();
    });

You could then return this new promise to whatever calling code would consume this:

var request = function () {
    return $.ajax()
        .then(function (response) {
            return $.Deferred(function (deferred) {
                var problem = hasError(response);

                if (problem) {
                    return deferred.reject(problem)
                }

                deferred.resolve(response);
            }).promise();
        });
};

request()
    .done(function (response) {
        // handle success
    })
    .fail(function (problem) {
        // handle failure
    });
Share:
15,437

Related videos on Youtube

Bergi
Author by

Bergi

ecmascript-6 and promise evangelist.

Updated on July 14, 2022

Comments

  • Bergi
    Bergi almost 2 years

    I have an $.ajax promise and want to check whether my (syntactically valid) response contains an error, triggering the rejected status in that case.

    I have worked with my own promise library which deals such tasks easily. I do not really like jQuery's Promise (cache) implementation with its Deferred object and may have overlooked something, because I seldom use it. I think the way to go is just using .then(), which seems to be rather complicated:

    return $.ajax(...).then(function success(response) {
        var problem = hasError(response);
        if (problem) {
            var error = new $.Deferred;
            error.reject(problem);
            return error;
        } else
            return response;
    });
    

    This should return a promise which is rejected in case of network errors or problems with the response. But is returning a rejected deferred really the [only|best|shortest] way to go?

    I also would appriciate help on how to deal with such "error-throwing response handlers" in the ajax options themselfes, I could not find good documentation about them.


    Disclaimer: No, I cant change the server responses. The problem-detecting method is synchronous. I don't want to use other libraries, I'm particularly interested in the way jQuery solves this.

  • Bergi
    Bergi almost 12 years
    What's the difference between pipe and then?
  • Eli
    Eli almost 12 years
    pipe allows you to filter a deferred's values before they are sent to any registered promise handlers. then is really just used to register handlers when a promise resolves or rejects.
  • Bergi
    Bergi almost 12 years
    No, by specification .then() would do exactly that. Or did jQuery threw this up and swapped method names?
  • Eli
    Eli almost 12 years
    jQuery Deferreds are not Promises/A compliant. They are a derivation of Promises/A and "Promises/J" jaubourg.net/38261410.
  • Bergi
    Bergi almost 12 years
    Uh, that's good to know. I will read through that article tomorrow.
  • Eli
    Eli about 11 years
    Sorry, there was an error in the code. It should now be fixed, and updated to work with jQuery 1.8+.
  • Eli
    Eli almost 11 years
    Note that jQuery is attempting to make Deferreds more like Promises/A.
  • Roamer-1888
    Roamer-1888 over 9 years
    @BenjaminGruenbaum, after editing, what are deferred and problem?
  • Benjamin Gruenbaum
    Benjamin Gruenbaum over 9 years
    Good catch, I was careless. Fixed.
  • Vladislav Rastrusny
    Vladislav Rastrusny about 9 years
    @Eli What are "Promises/J"? I can't find any references. jaubourg.net looks empty of content.
  • Matthijs Kooijman
    Matthijs Kooijman over 5 years
    Responding to the first comment: It seems that .then() was changed to behave like pipe. The jQuery docs say: "As of jQuery 1.8, the deferred.then() method returns a new promise that can filter the status and values of a deferred through a function, replacing the now-deprecated deferred.pipe() method."