Declarative HTTP requests for React.
Making a single HTTP request is not difficult to do in JavaScript. However, complex web applications often make many requests as the user navigates through the app.
Features such as request deduplication and response caching can often save the developer of apps like these from headache and bugs. Although it is possible to implement these features imperatively, it requires that you write a bit of code, and that code can be tedious to test.
A declarative API makes things a lot simpler for you, which is where React Request comes in. React Request is a backend-agnostic, declarative solution for HTTP requests in React, and its deduping and caching features are a delight to use.
✓ Uses the native fetch API
✓ Smart deduping of requests
✓ Customizable response caching
✓ Provides hooks to integrate with external stores (such as Redux)
✓ Reasonable footprint (~2kb gzipped)
Install using npm:
npm install react-request
or yarn:
yarn add react-request
Here's a quick look at what using React Request is like:
import React, { Component } from 'react';
import { Fetch } from 'react-request';
class App extends Component {
render() {
return (
<Fetch url="https://jsonplaceholder.typicode.com/posts/1">
{({ fetching, failed, data }) => {
if (fetching) {
return <div>Loading data...</div>;
}
if (failed) {
return <div>The request did not succeed.</div>;
}
if (data) {
return (
<div>
<div>Post ID: {data.id}</div>
<div>Post Title: {data.title}</div>
</div>
);
}
return null;
}}
</Fetch>
);
}
}
Need to make multiple requests? You can use any tool that you would like that allows you to "compose" render prop components together. This example uses React Composer:
import React, { Component } from 'react';
import Composer from 'react-composer';
class App extends Component {
render() {
return (
<Composer
components={[
<Fetch url="https://jsonplaceholder.typicode.com/posts/1" />,
<Fetch
url="https://jsonplaceholder.typicode.com/posts/1"
method="DELETE"
/>
]}>
{([readPost, deletePost]) => {
return (
<div>
{readPost.fetching && 'Loading post 1'}
{!readPost.fetching && 'Post 1 is not being fetched'}
<button onClick={() => deletePost.doFetch()}>
Delete Post 1
</button>
</div>
);
}}
</Composer>
);
}
}
These examples just scratch the surface of what React Request can do for you. Check out the API reference below, or read the guides, to learn more.
A component for making a single HTTP request. It accepts every value of init
and input
from the
fetch()
API as a prop, in addition to a few other things.
The props that come from the fetch()
method are:
url
method
: defaults to"GET"
body
credentials
headers
mode
cache
redirect
referrer
: defaults to"about:client"
referrerPolicy
: defaults to""
integrity
: defaults to""
keepalive
signal
To learn more about the valid options for these props, refer to the
fetch()
documentation.
The following example demonstrates some of the most commonly-used props that come from fetch()
:
<Fetch
url="/posts/2"
method="patch"
credentials="same-origin"
headers={{
'csrf-token': myCsrfToken
}}
body={JSON.stringify({ title: 'New post' })}>
{({ doFetch }) => {
<button onClick={() => doFetch()}>Update Post</button>;
}}
</Fetch>
In addition to the fetch()
props, there are a number of other useful props.
The render prop of this component.
It is called with one argument, result
, an object with the following keys:
fetching
: A Boolean representing whether or not a request is currently in flight for this componentfailed
: A Boolean representing whether or not the request failed for any reason. This includes network errors and status codes that are greater than or equal to400
.error
: An error object representing a network error occurred. Note that HTTP "error" status codes do not cause errors; only failed or aborted network requests do. For more, see the "Using Fetch" MDN guide.response
: An instance of Response. Thebody
will already be read, and made available to you asresponse.data
.data
: The data returned inresponse
. This will be different fromresponse.data
if atransformData
prop was passed to<Fetch/>
.doFetch
: A function that makes the HTTP request. See notes below.url
: The URL that was passed into<Fetch />
.requestName
: The name of the request (seerequestName
below)requestKey
: The computed request key
There are three common use cases for the doFetch
prop:
- You can use it to "refresh" the data by making a follow-up request for read requests
- You can use it to retry the request if there is any sort of error
- You must manually call this method to actually make the request anytime that the
lazy
prop is passed astrue
.
doFetch
accepts one argument: options
. Any of the fetch()
options, such as url
, method
, and
body
are valid options
. You may also specify a new requestKey
if you are manually generating your
own keys. This method allows you to customize the request from within the component based on the
component's state.
Whether or not the request will be called when the component mounts. The default value is based on the request method that you use.
Method | Default value |
---|---|
GET, HEAD, OPTIONS | false |
POST, PUT, PATCH, DELETE | true |
<Fetch url="/books" lazy>
{({ doFetch }) => {
<button onClick={() => doFetch()}>Load the books</button>;
}}
</Fetch>
A function that is called just before a network request is initiated. It is called with one argument, an object with the following keys:
url
: The URL of the requestinit
: The second argument passed toglobal.fetch()
, which specifies things such as the body, method, and so onrequestKey
: Either the computed request key, or the value of therequestKey
prop
This feature is useful for analytics, or syncing response data with a data store such as Redux.
Note: This function is not called when the component reads from the cache.
A function that is called anytime that a network response is received. It is called with one argument, an object with the following keys:
url
: The URL of the requestinit
: The second argument passed toglobal.fetch()
, which specifies things such as the body, method, and so onrequestKey
: Either the computed request key, or the value of therequestKey
propresponse
: The response that was received from the HTTP requestdata
: The transformed data from the response. This will be different fromresponse.data
if atransformData
function was passed as a prop to<Fetch/>
.error
: An error returned from the HTTP requestdidUnmount
: A Boolean representing whether or not the component has unmounted
This can be used for analytics or syncing response data with a data store such as Redux.
Note: This function is not called when the component reads from the cache.
A function that is called every time a response is received, whether that
response is from the cache or from a network request. Receives two arguments:
error
and response
.
<Fetch
url="/posts/2"
onResponse={(error, response) => {
if (error) {
console.log('Ruh roh', error);
} else {
console.log('Got a response!', response);
}
}}>
{() => {
<div>Hello</div>;
}}
</Fetch>
A function that is called with the data returned from the response. You can use this
hook to transform the data before it is passed into children
.
<Fetch
url="/posts/2"
transformData={data => data.post>
{({ fetching, error, response, data }) => {
<div>
{fetching && ('Loading...')}
{error && ('There was an error.')}
{!fetching && !error && response.status === 200 && (
<div>
<h1>{data.title}</h1>
<div>{data.content}</div>
</div>
)}
</div>
}}
</Fetch>
Note:
transformData
does not modify the value ofresponse.data
. The transformed data is made available to you in the render prop argument under thedata
key.
The content type of the response body. Defaults to "json"
unless the response has a 204 status code,
in which case it will be "text"
instead. Valid values are any of the methods on
Body.
Alternatively, you may specify a function that returns a string. The function will be called with one
argument: response
. This allows you to dynamically specify the response type based on information
about the response, such as its status code.
// If you have an endpoint that just returns raw text, you could, for instance, convert it into
// an object using `responseType` and `transformData`.
<Fetch
url="/countries/2"
responseType="text"
transformData={countryName => {
return {
countryName
};
}}>
{({ data }) => {
if (data) {
return <div>{data.countryName}</div>;
}
return null;
}}
</Fetch>
// Some "enterprisey" endpoints return text stack traces anytime that they error. A function
// `responseType` can protect you against this.
<Fetch
url="/countries/2"
responseType={response => {
// Only parse as JSON when the request's code is not an error code, and it is
// not 204 No Content.
return response.ok && response.status !== 204 ? 'json' : 'text';
}}
transformData={countryName => {
return {
countryName
};
}}>
{({ data }) => {
if (data) {
return <div>{data.countryName}</div>;
}
return null;
}}
</Fetch>
A name to give this request, which can help with debugging purposes. The request name is analogous to a function name in JavaScript. Although we could use anonymous functions everywhere, we tend to give them names to help humans read and debug the code.
<Fetch url={`/posts/${postId}`} requestName="readPost" />
Note: This feature is analogous to the operation name in GraphQL.
This determines how the request interacts with the cache. Valid options are:
"cache-first"
"cache-and-network"
"network-only"
"cache-only"
For documentation on what each of these values do, refer to the response caching guide.
The default value of this prop is based on the value of the method
prop that you pass to <Fetch/>
.
Method | Default value |
---|---|
GET, HEAD, OPTIONS | "cache-first" |
POST, PUT, PATCH, DELETE | "network-only" |
This prop behaves identically to the Apollo prop with the same name.
Whether or not the response will be cached. The default value is based on the value of the method
prop that you pass
to <Fetch/>
.
Method | Default value |
---|---|
GET, HEAD, OPTIONS | true |
POST, PUT, PATCH, DELETE | false |
For documentation on this prop, refer to the response caching guide.
A Boolean value representing whether or not the request should be
deduplicated.
Defaults to true
.
A string that is used to control the request deduplication and response caching features. By default, a key is generated for you. Specifying a custom key is an advanced feature that you may not need.
For more, see the request key guide.
The rest of the API documentation describes the other named exports from the react-request
package.
This is the fetchDedupe
export from the Fetch Dedupe
library. Fetch Dedupe powers the request deduplication in React Request.
If, for whatever reason, you need to make a standalone HTTP request outside of the
<Fetch />
component, then you can use this with confidence that you won't send a
duplicate request.
For more, refer to the documentation of fetch-dedupe.
Generates a request key. All of the values are optional.
This method comes from fetch-dedupe
.
Return a Boolean representing if a request for requestKey
is in flight or not.
This method comes from fetch-dedupe
.
Wipes the cache of deduped requests. Mostly useful for testing.
This method comes from fetch-dedupe
.
Note: this method is not safe to use in application code.
Wipes the cache of cached responses. Mostly useful for testing.
Note: this method is not safe to use in application code.
This library was inspired by Apollo. The library Holen was referenced during the creation of this library.