Ecosyste.ms: Awesome

An open API service indexing awesome lists of open source software.

Awesome Lists | Featured Topics | Projects

https://github.com/lddubeau/bluejax

jQuery AJAX wrapped in Bluebird promises.
https://github.com/lddubeau/bluejax

ajax bluebird diagnosis jquery promise retries

Last synced: about 1 month ago
JSON representation

jQuery AJAX wrapped in Bluebird promises.

Awesome Lists containing this project

README

        

Bluejax is a library that wraps jQuery's
[``ajax``](https://api.jquery.com/jquery.ajax/) function in [Bluebird
promises](http://bluebirdjs.com/docs/getting-started.html). This is not the
first library of this kind, but the other libraries that existed when Bluejax
was created did not satisfy our needs, or appeared defunct.

Features
========

* Wraps ``jQuery.ajax`` in Bluebird promises.

* Optionally retry failed queries a number of times before giving up.

* Optionally diagnoses failed queries: network failure, server down, something
else?

jQuery Versions Supported
=========================

In the 3.x and 2.x series: any version.

In the 1.x series: 1.11 or later. This being said, Bluejax probably works fine
with 1.9 and 1.10.

Platforms Supported
===================

Bluejax's is tested on Chrome, Firefox, IE11 and 10, Edge, Opera, and Safari (on
El Capitan, Yosemite, Mavericks). We test against the latest versions offered by
the vendors of these browsers on their respective platforms.

Bluejax's suite fails on IE9. I currently have no plans to work on adding
support for IE9 but if you want to provide a pull request that will make Bluejax
run on IE9 and will keep its current tests passing, you are welcome to do so.

Loading Bluejax
===============

AMD
---

Your loader should be configured so that it can find Bluejax, jQuery
and Bluebird. jQuery and Bluebird are requested by Bluejax as
``jquery`` and ``bluebird`` respectively.

CommonJS
--------

Once installed, you should just be able to do ``var bluejax =
require("bluejax")``. jQuery and Bluebird should also be installed. Just like in
the AMD case, they are required as ``jquery`` and ``bluebird`` respectively.

``script`` elements
-------------------

If you are just loading it in a browser with ``script``. jQuery and
Bluebird must have been loaded beforehand.

Using Bluejax
=============

The module exports these items:

* ``ajax(...)`` is a function that passes all its arguments to
``jQuery.ajax``. By default, it returns a promise that resolves to the data
received. If it fails, it will reject the promise with a ``GeneralAjaxError``
or a an exception whose class is derived from ``GeneralAjaxError``.

ajax(url).then(function (data) {
document.getElementById("foo").innerHTML = data;
});

* ``ajax$(...)`` does the same thing as ``ajax(...)`` but it returns an object
that has the keys ``xhr`` and ``promise``. The value of ``promise`` is the
same as the value returned by ``ajax(...)`` the ``xhr`` object is an object
like the ``jqXHR`` returned by ``jQuery.ajax``.

**Important note:** the ``xhr`` object only encapsulate **the tries** that
Bluejax performs. It will be successful if the Ajax query was successful
within the number of tries specified and will fail if the tries failed. It is
**not** affected by the diagnosis done after all tries failed. If you need
diagnostic information you **must** use the promise.

This call can be useful for the purpose of inserting Bluejax in contexts that
expect to work with ``jQuery.ajax``. For instance, I can make Backbone use
Bluejax for all its Ajax requests. In this case, I use a special
``useBluejax`` option to turn on the use of Bluejax. Those calls that do use
use ``useBluejax`` go through unchanged.

var origAjax = Backbone.ajax;

Backbone.ajax = function ajaxWrapper(options) {
if (!options.useBluejax) {
return origAjax.call(this, options);
}

// Return the xhr because this is what callers to ``Backbone.ajax``
// expect.
return ajax$(options).xhr;
};

Although the callers of ``Backbone.ajax`` do not get a promise. The callers
still benefit from the possibility of retrying queries. If everything goes to
hell they do not get the diagnosis information themselves but a handler that
listens for unhandled rejections could use the rejection information to
provide an informative error message. This is actually the case for my
Backbone application that uses the snippet above.

* ``make(options, field)`` is a utility function that creates a new
``ajax$``-like function. The ``options`` parameter is an object containing
Bluejax options. The returned value is a new function that works like ``ajax``
but which has its Bluejax options set to the values contained in the
``options`` object.

The ``field`` parameter allows you to automatically extract a field. If you do
not specify a value for ``field``, then the return value will be the same
object returned by ``$ajax``. If you want to retrieve only one field from that
object, you must specify the field name. To get the same value as the
``ajax(...)`` call, you'd need to put ``"promise"`` for the value of
``field``.

Example: it is possible to create a new function that will return verbose
results: ``var ajaxVerbose = make({verboseResults: true}, "promise");`` and
use it in the same way ``ajax`` is used:

ajaxVerbose("http://example.com").spread(function (data, textStatus,
jqXHR) {
...
});

* ``GeneralAjaxError`` is a class that derives from JavaScript's stock
``Error``. It aims to provide a somewhat saner way to handle Ajax errors than
what jQuery provides by default. When ``jQuery.ajax`` fails, Bluejax creates
an exception derived from ``GeneralAjaxError`` that has its ``jqXHR``,
``textStatus`` and ``errorThrown`` fields set to the corresponding fields of
callback that should be passed to the ``.fail(...)`` method of the object
returned by ``jQuery.ajax``. Its ``message`` field is constructed as follows:

* If ``errorThrown`` is set, the message is "Ajax operation
failed: ``errorThrown`` (``jqXHR.status``).".

* Otherwise, if ``textStatus`` is set, the message is "Ajax
operation failed: ``textStatus``.".

* Otherwise, the message is "Ajax operation failed."

* ``HttpError`` is raised if the response had an HTTP status that signaled an
error.

* ``TimeoutError`` is raised if the rejection was caused by a timeout.

* ``AbortError`` is raised if the rejection was caused by an abort.

* ``ParserError`` is raised if the rejection was caused by a parsing problem.

* ``ConnectivityError`` indicates a network problem. This class of error is
never raised directly but is raised through its children:

+ ``BrowserOfflineError`` is raised if the browser is offline.

+ ``ServerDownError`` is raised if the server is down.

+ ``NetworkDownError`` is raised if the network is down.

* If none of the more specialized cases above apply, then ``AjaxError`` is
raised.

Options
-------

Bluejax currently supports these options:

* ``tries`` tells Bluejax to retry the query for a number of times. Note that
the value here should be a number greater than 1. (Values less than 1 yield
undefined behavior.)

* ``shouldRetry`` is a function with the following signature
``shouldRetry(jqXHR, textStatus, errorThrown)``. It should return ``true`` if
the query should be retried, or ``false`` if an error should be returned
immediately.

If no value is specified, the default function returns ``true`` if the
previous query failed due to reasons **other** than the HTTP status code
reporting an error, aborted or had a parser error. Basically, it retries the
connection if the issue appears to be at the network level rather than an
application issue.

* ``delay`` specifies the delay between retries, in milliseconds.

* ``diagnose`` is an object with the following keys:

* ``on`` must be ``true`` for diagnosis to happen. (This makes diagnosis
easy to turn off, while keeping the other diagnosis settings intact.)
**Make sure to read the section on diagnosis rules and URL transformations
below before to understand what it is that happens when you turn diagnosis
on!!!**

* ``serverURL`` must be a URL that used to test whether your server is
running or not. We recommend making it a path that is inexpensive to
serve. For instance, your internet-facing nginx instance could have a rule
that serves 200 and no contents for GETs to ``/ping``. Bluejax uses this
URL to double check whether your server is up.

* ``knownServers`` must be an array of URLs to known internet
servers. Bluejax uses these URLs to determine whether the Internet is
accessible or not.

* ``verboseExceptions`` when set to ``true`` will cause exception messages to
additionally contain the text "Called with: " followed by a JSON dump of the
options that were passed to ``ajax(...)``. This can be useful to identify
which call precisely is failing.

* ``verboseResults`` causes the promise to resolve to ``[data, textStatus,
jqXHR]`` where each element of the array is the corresponding parameter passed
to the callback of the ``.done(...)`` method of the object returned by
``jQuery.ajax``. You would use this in a case where just receiving ``data`` is
not enough for your usage scenario.

Bluebird's ``.spread`` method is useful to unpack the array:

ajax(url, {
bluejaxOptions: { verboseResults: true }
}).spread(function (data, textStatus, jqXHR) {...

There are two ways to set Bluejax options:

* You can set set the ``bluejaxOptions`` field on a settings object passed to
``ajax``. Remember that ``ajax(...)`` takes the same parameters as
``jQuery.ajax``. When you pass a ``settings`` parameter to the call, it may
contain a ``bluejaxOptions`` field that sets ``verboseExceptions``:

bluejax.ajax({
url: "http://example.com",
bluejaxOptions: {
verboseExceptions: true
}
});

* You can create a new ``ajax``-like function with ``make``.

Diagnosis Rules
===============

Diagnosis happens only if the final try for the request failed with an error
other than an HTTP error, an abort or a parser error. Otherwise, no diagnosis
occurs and the error is reported immediately.

Bluejax uses the following rules when diagnosis is requested:

1. If ``navigator.onLine`` is false, Bluejax reports that the browser is
offline.

2. If a ``serverURL`` is specified, then it checks whether the server is
responds to a GET at this URL:

A. If the server responds, Bluejax reports the error that was reported by the
last try.

B. If the server does not respond, Bluejax reports the result of a
connectivity check.

3. If a ``serverURL`` was not specified, Bluejax reports the result of a
connectivity check.

Connectivity Check
------------------

1. If ``knownServers`` does not exist or is an empty list, then it reports that
the server appears to be down.

2. Otherwise, Bluejax contacts all the servers. If none of them respond, then it
reports that the network appears to be down. Otherwise, it reports that the
server appears to be down.

URL Checking Rules
------------------

For all URLs used in diagnosis, these two transformations are applied in order:

1. If the URL ends with a `/` and has no query string, then the URL is
transformed by adding `favicon.ico`. Requesting the root page of a site may
result in a large amount of data being returned. Requesting `favicon.ico`
would in most cases result in a relatively small amount of data.

2. If the URL has no query, then the URL is transformed by adding a query that
is a single number corresponding to the current time. This is done to bust
caches.

So if you specify a known server as ``http://www.google.com/`` the URL used for
the query will be ``http://www.google.com/favicon.icon?ttttt``, where ``ttttt``
is the number described above. If you specify ``http://www.example.com/foo``
then the URL used for the query will be ``http://www.example.com/foo?ttttt``. If
the URL you give in the options contains a query, it won't be modified **at
all**. (We do not recommend such case.)

Developing Bluejax
==================

If you produce a pull request run ``gulp lint`` and ``gulp test`` first to make
sure they run clean. If you add features, do add tests.

Coverage
--------

We need a Mocha run to test loading Bluejax as a CommonJS module with ``script``
elements. The Karma run, which exercises over 95% of the code, uses RequireJS
to load Bluejax.

Ideally, we combine the results of the Karma runs with the result of the Mocha
run. The problem though is that as we speak, ``karma-coverage`` uses Istanbul
0.4.x but to get coverage with Mocha with code that has run through Babel, we
need Istanbul 1.0.0-alpha2 or higher. We've not been able to combine the formats
produced by the various versions.

Testing
-------

[![Browser Stack](https://www.browserstack.com/images/mail/browserstack-logo-footer.png)](https://www.browserstack.com)

Bluejax is tested using
[BrowserStack](https://www.browserstack.com). BrowserStack provides this service
for free under their program for supporting open-source software.