Redux Easy Async makes handling asynchronous actions, such as API requests, simple, reliable, and powerful.
Redux is fantastic tool for managing application state but since actions, by their very nature, are synchronous, asynchronous requests (such as to APIs) can be more challenging. Standard approaches involve a lot of boilerplate, reinventing the wheel, and opportunity for error. For more info see: Motivation
Redux Easy Async provides a simple yet powerful approach for creating and tracking asynchronous actions. Features:
- Create asynchronous actions (including associated constants for start, success, and fail) with a single call.
- Create reducer(s) that automatically track the status of asynchronous actions,
i.e. pending, completed. No more dispatching loading actions or storing
isFetching
state. - Optional configuration to parse API responses pre-reducer, conditionally make requests, prevent multiple requests, and more.
- Makes all your dreams come true and turns you into the person you always wanted to be.
With NPM:
npm install redux-easy-async --save
or with Yarn:
yarn add redux-easy-async
-
Add the async middleware to wherever you create your redux store:
// assuming have created and imported a root reducer: // e.g. import rootReducer from './reducers'; import { createStore, applyMiddleware } from 'redux'; import { createAsyncMiddleware } from 'redux-easy-async'; const asyncMiddleware = createAsyncMiddleware(); createStore(rootReducer, applyMiddleware(asyncMiddleware));
-
Create an async action:
import { createAsyncAction } from 'redux-easy-async'; export const fetchUser = createAsyncAction('FETCH_USER', (id) => { return { makeRequest: () => fetch(`https://myapi.com/api/user/${id}`), }; }); // fetchUser is now a function that can be dispatched: // const id = 1; // dispatch(fetchUser(id)) // // but it also has constant attached for start, success, fail // fetchUser.START_TYPE === "START_FETCH_USER" // fetchUser.SUCCESS_TYPE === "SUCCESS_FETCH_USER" // fetchUser.FAIL_TYPE === "FAIL_FETCH_USER"
-
Handle the action in a reducer somewhere:
const usersReducer = (state = {}, action) => { const { type, payload } = action; switch (type) { case fetchUser.SUCCESS_TYPE: // do something here on succcess case fetchUser.FAIL_TYPE: // do something here with fail default: return state; } };
-
Dispatch the action somewhere in your app:
dispatch(fetchUser(1));
-
Profit!
-
add a requests reducer that will track all async actions:
import { createAsyncReducer } from 'redux-easy-async'; // createAsyncReducer takes an array of async actions and returns // a reducer that tracks them const requestsReducer = createAsyncReducer([fetchPost]); // now you have a reducer with keys for each each action passed to it // { // FETCH_USER: { // hasPendingRequests: false, // pendingRequests: [], // } // }
-
Add the requests reducer to your main reducer somewhere:
import { combineReducers } from 'redux'; const rootReducer = combineReducers({ requests: requestsReducer, // ...the other reducers for your app here });
-
Show loading spinner in a component somewhere:
// assuming you have the state of the request store, e.g: // const requests = store.getState().requests; // -- or -- // const { requests } = this.props; // const isLoading = requests.FETCH_USER.hasPendingRequests; return ( <div> { isLoading && <div>Show a loading spinner</div> } { !isLoading && <div>Show user data</div> } </div>);
The examples directory includes fully working examples which you can run locally and test out Redux Easy Async.
-
Install dependencies for the example you want to try (with npm or yarn):
> cd examples/basic > npm install (or yarn install)
-
Start the server
> npm start (or yarn start)
-
Go to
http://localhost:4001/
in your browser.
- http://redux.js.org/docs/advanced/AsyncActions.html
- loading or not loading should just be a status not a seperate loading action
Kind: global function
Returns: function
- actionCreator
Param | Type | Default | Description |
---|---|---|---|
type | string | object |
can either be a string (e.g. "GET_POSTS") or a a constants object created with createAsyncConstants. | |
fn | function |
action creator function that returns an object with action configuration. See example below for configuration options. Only makeRequest is required . |
|
[options] | Object |
additional configuration options | |
[options.namespace] | Object |
REDUX_EASY_ASYNC_NAMESPACE |
the middleware action type this action will be dispatched with. You most likely don't want to modify this unless for some reason you want multiple instances of async middleware. |
Example (All configuration options for async action)
import { createAsyncAction } from 'redux-easy-async';
const myAction = createAsyncAction('MY_ACTION', () => {
return {
// function that makes the actual request. Return value must be a promise. In this example
// `fetch()` returns a promise. **REQUIRED**
makeRequest: () => fetch('/api/posts'),
// *OPTIONAL*
// additional meta that will be passed to the start, success, and fail actions if any. All
// actions will have the following meta:
// - `actionName`
// - `asyncType`("start", "success", or "fail")
// - `requestStartTime`
// - `asyncID`: an unique id for each request
// Success and fail actions will additionally have:
// - `requestDuration`
// - `resp`: the raw api response. Because of the nature of the promises errors that
// cause the makeRequest promise to be rejected will also get caught here as `resp`
// and cause a failed request action.
meta = {},
// function that takes your redux state and returns true or false whether to proceed with
// the request. For example: checking if there is already a similar request in progress or
// the requested data is already cached. *OPTIONAL*
shouldMakeRequest = (state) => true,
// `parseStart`, `parseSuccess`, and `parseSuccess` can be useful if you want to modify
// raw API responses, errors, etc. before passing them to your reducer. The return value
// of each becomes the payload for start, success, and fail actions. By default response
// will not be modified.
//
// the return value of `parseStart` becomes the payload for the start action. *OPTIONAL*
parseStart = () => null,
// the return value of `parseSuccess` becomes the payload for the success action. *OPTIONAL*
parseSuccess = resp => resp,
// the return value of `parseFail` becomes the payload for the fail action. *OPTIONAL*
parseFail = resp => resp,
}
})
Creates an object with constant keys NAME
, START_TYPE
, SUCCESS_TYPE
, FAIL_TYPE
in the
format that createAsyncAction, createAsyncReducer, and
createAsyncReducer accept. Note: this is an extra optional step for those that
prefer to separate action creator definitions from constants. If you don't know/case then just
createSingleAsyncAction.
Kind: global function
Returns: object
- returns an object with keys: NAME
, START_TYPE
, SUCCESS_TYPE
, and
FAIL_TYPE
Param | Type | Description |
---|---|---|
type | string |
the base name for this constant, e.g. "GET_USER" |
Example
const GET_USER = createAsyncConstants('GET_USER');
// {
// NAME: 'GET_USER', // general name for action
// START_TYPE: 'START_GET_USER', // start type of the this async action
// SUCCESS_TYPE: 'SUCCESS_GET_USER', // success type of the this async action
// FAIL_TYPE: 'FAIL_GET_USER' // fail type of the this async action
// }
Creates an instance of middleware necessary to handle dispatched async actions created with createAsyncAction.
Kind: global function
Returns: function
- redux middleware for handling async actions
Param | Type | Default | Description |
---|---|---|---|
options | object |
options to create middleware with. | |
[options.requestOptions] | object |
{} |
options that will be passed to all action's makeRequest functions: e.g. makeRequest(state, requestOptions) . |
[options.namespace] | string |
"REDUX_EASY_ASYNC_NAMESPACE" |
the action type the middleware will listen for. You most likely don't want to modify this unless for some reason you want multiple instances of async middleware. |
Example
import { createAsyncMiddleware } from 'redux-easy-async';
const asyncMiddleware = createAsyncMiddleware();
...
// Now add to your middlewares whereever your store is created.
// Typically this looks something like:
// const middlewares = [asyncMiddleware, ...other middlewares]
Creates a requests reducer that automatically tracks the status of one or more async actions created with createAsyncAction. As you dispatch async actions this reducer will automatically update with number of requests, request meta for each, a boolean for whether any requests are currently pending for this request.
Kind: global function
Returns: function
- Redux reducer automatically tracking the async action(s)
Param | Type | Description |
---|---|---|
types | Array | String | Object | function |
one or more async actions to track. Either a single instance or an array of one or more of the following: a string (e.g. `"GET_POSTS"``), a constants object created with createAsyncConstants, or an async action created with createAsyncAction. Typically you will want to pass an array of actions to track all async actions for your application in one place. |
Example
import { createAsyncAction, createAsyncConstants } from '`redux-easy-async';
// Types can async action, constants object, or string:
// string constant
const FETCH_POSTS = 'FETCH_POSTS';
// async action
export const fetchUser = createAsyncAction('FETCH_USER', (id) => {
return {
makeRequest: () => fetch(`https://myapi.com/api/user/${id}`),
};
});
// async constant
const fetchComments = createAsyncConstants('FETCH_COMMENTS');
// now we can create a reducer from the action or constants we've defined
const requestsReducer = createAsyncReducer([FETCH_POSTS, fetchUser, fetchComments]);
// Now `requestsReducer` is reducer that automatically tracks each of the asynchronous action
// types. It's state looks something like this to start:
// {
// {
// FETCH_POSTS: {
// hasPendingRequests: false,
// pendingRequests: [],
// },
// {
// FETCH_USER: {
// hasPendingRequests: false,
// pendingRequests: [],
// }
// {
// FETCH_COMMENTS: {
// hasPendingRequests: false,
// pendingRequests: [],
// }
// }
Author: Evan Hobbs - NerdWallet
License: MIT - LICENSE
for more information.
Todo
- images/logo
- finish Motivation section.
- add error logging in middleware?
- add more complicated example