Many applications transfer data using HTTP requests. One feature of HTTP requests is that they can be cancelled.
As a rule of thumb, for each request that you make in your application, there will always be one situation where you will want to cancel that request to prevent bugs.
Canceling requests prevents race conditions. A race condition occurs when you have two requests in flight at the same time, but your code is written such that one must resolve before the other. As you may know, requests can resolve in any order, which leads to the bugs.
A common pattern we've seen in Redux applications is this:
- whenever a request completes, an action is dispatched
- a reducer responds to that action, updating the state
- a view renders using the latest data from the state
This pattern isn't bad – it's great! You just need to be aware that it can lead to bugs when requests aren't cancelled.
Let's look at two examples that demonstrate the problem.
Consider a "typeahead" component that allows a user to type to search for books. Even when the user's input is debounced, there are situations where two (or more) active requests can be in flight. The user can type, wait a moment for a request to send, then type one key to fire off a second request. If the backend response time is slower than the debounce time, then two search requests will be in flight at once.
Because the requests can complete in any order, it is possible that the user's earliest search results will return after their latest search results. If your code is written such that the last-received search results are displayed, then that can lead to the wrong search results being shown.
Canceling a previous request resolves this problem, as it means that only one search request will ever be in flight at a time, and it will be the user's most recent search. If you're using an action creator that returns a native XMLHttpRequest object, such as the action creator from Redux Resource XHR, then your code may look like this:
class Typeahead extends Component {
searchBooks: function(query) {
if (this.searchBooksRequest) {
this.searchBooksRequest.abort();
}
this.searchBooksRequest = this.props.searchBooks(query);
}
}
Another tip is to cancel any requests when the component unmounts. If the search
results aren't needed when the component unmounts, then letting a request complete
will still update your state tree, which can cause unnecessary renders in your
application. Canceling requests in componentWillUnmount
might look something
like:
class Typeahead extends Component {
componentWillUnmount: function() {
if (this.searchBooksRequest) {
this.searchBooksRequest.abort();
}
}
}
Consider a page of an application that displays books. Often, long lists of resources will be paginated, and a user can move between the pages by clicking a "next" or "previous" button.
A race condition can occur when the user clicks the buttons too fast, or when the backend service is slow. If your view layer renders out the results of the latest response, then it's possible that they could see the results from the wrong page.
Similar to the search example above, ensuring that only a single page of data is being fetched at once is the solution.
The solution is the same as the typeahead example:
- cancel any existing page-change requests before starting a new one
- consider if it makes sense to cancel the request when the component unmounts
A common feature of these bugs is that they depend on unreliable network conditions, so they don't usually come up in a development environment. This makes them easy to ignore, but they're still worth protecting against.
Typically, applications do not need to inform the user when a request is aborted.
Accordingly, Redux Resource does not track if a request is in an aborted state. Instead,
we encourage you to set the request status back to "IDLE"
when the request is canceled.
For a read request, this may look something like:
import { actionTypes } from 'redux-resource';
// You will need to determine that the request was aborted;
// different libraries have different systems for doing this
let requestWasAborted;
if (requestWasAborted) {
dispatch({
type: actionTypes.READ_RESOURCES_IDLE,
...otherActionAttributes
})
}
Note: If your application requires tracking the aborted status of a request, you can write a plugin to add support for additional action types.
Note: We understand that some users want their action type names to reflect the action that is being performed, rather than the result of the action. We agree that this is a good practice to follow. If you do, too, it may irritate you that there is no
READ_RESOURCES_ABORT
action type. This is omitted in an effort to keep the surface area of Redux Resource small, since that action would behave the same asREAD_RESOURCES_IDLE
.
There are many different tools developers use to make requests. In this section, we will go through how to cancel requests using some of the most common tools.
In browsers, the native way to cancel a request is to call the abort
method
on an XMLHttpRequest.
const myRequest = new XMLHttpRequest();
myRequest.open('GET', 'books');
myRequest.abort();
The xhr library simplifies the creation of
XHR objects. Because it returns a native XMLHttpRequest object, canceling
requests with xhr
is the same as when you use the native XMLHttpRequest
constructor.
import xhr from 'xhr';
const request = xhr.get('/books/23', (err, res) => {
if (req.aborted) {
console.log('Request cancelled');
}
});
request.abort();
Redux Resource XHR for
Redux Resource uses xhr
for requests.
The action creator exported by this library returns a native XHR object, so you
can use the abort
method to cancel those requests.
The native fetch
method is a tool for making requests that returns a
Promise.
Native Promises cannot be cancelled (yet),
but you can get around this limitation by "ignoring" the server response.
One way to do this is to create a function that you return from your action creators. Then, only fire the "success" action as long as that function is not called.
Although there are benefits to actually canceling the request, this solution will avoid the race condition bugs described in this guide.
axios is a Promise-based approach to HTTP requests that supports cancellation. It uses a system of canceling Promises called "Cancel Tokens," which is based off of a withdrawn cancelable Promises proposal.
To see how to cancel axios requests, refer to the axios documentation.
Bluebird is Promise implementation that supports
cancellation using an onCancel
method.
Refer to the Bluebird documentation
for specifics on the onCancel
method.
RxJS is a Reactive Programming library for async code using Observables. It includes Observable.ajax()
for HTTP requests and supports cancellation by calling subscription.unsubscribe()
, or using something like .takeUntil()
.
If you happen to be using redux-observable, refer to that library's documentation for a recipe for request cancellation.
You don't always need to cancel requests. Two other options are:
-
Prevent the user from performing an action more than once. For instance, in the search example, you could prevent the user from typing if a search of theirs is already in flight. And in the pagination example, you can disable the "next" and "previous" buttons until the new page loads. In these examples, this approach is clearly a subpar experience, but sometimes it does make sense. For instance, disabling a "Delete" button for a book once the user clicks it once, or disabling a "Buy" button after the user confirms a purchase.
-
You could write code to keep track of which request's response is the correct one to display. We generally find it to be simpler to cancel requests instead.
There may be other options, too, and different approaches may make more sense to you based on the situation.