This is an example of using redux-logic for async fetching with RxJS ajax triggered by a USERS_FETCH
action type and using the new processOptions
feature introduced in [email protected]. RxJS ajax has XHR abort on cancel so in-flight requests are immediately cancelled by redux-logic.
The new processOptions
feature allows you to clean up your code even further by declaring that you want to use the return value and apply these action types. Then you can simply return a promise, observable, object, or error and redux-logic will dispatch the associated type.
This also cleans up the isolated test code since you can simply test the resultant promise.
It showcases some of the declarative functionality built into redux-logic, so simply by specifying a cancelType and setting latest to true, we enable this code to be cancellable and also enable it to only take the latest request if multiple come in. No code had to be written by us to leverage that functionality.
Finally we are also showcasing that runtime dependencies can be injected rather than hard coded into your logic. So while I could have used RxJS ajax directly in this code, by injecting it I can now easily mock it out when testing.
// in src/users/logic.js
export const usersFetchLogic = createLogic({
type: USERS_FETCH,
cancelType: USERS_FETCH_CANCEL,
latest: true, // take latest only
processOptions: {
dispatchReturn: true, // use values from returned obs
successType: usersFetchFulfilled,
failType: usersFetchRejected
},
// use RxJS ajax injected as httpClient from configureStore logic deps
// we also have access to getState and action in the first argument
// but they were not needed for this particular code
process({ httpClient }) {
// the delay query param adds arbitrary delay to the response
// since dispatchReturn is true and successType and failType set
// our code becomes very clean, just return the observable
return httpClient.getJSON(`http://reqres.in/api/users?delay=${delay}`)
.map(payload => payload.data); // use data property of payload
}
});
-
src/configureStore.js - logicMiddleware is created with the combined array of logic for the app. Also the dependencies are defined that will be made available to all logic, so RxJS ajax is defined as httpClient.
-
src/rootLogic.js - combines logic from all other parts of the app and defines the order they appear in the logic pipeline. Shows how you can structure large apps to easily combine logic.
-
src/users/logic.js - the logic specific to the users part of the app, this contains our async fetch logic
-
src/users/actions.js - contains the action creators
-
src/users/reducer.js - contains a reducer which handles all the users specific state. Also contains the users related selectors. By collocating the reducer and the selectors we only have to update this one file to change the shape of our reducer state.
-
src/users/component.js - Users React.js component for displaying the status, fetch + cancel buttons, and the list of users
-
src/App.js - App component which uses redux connect to provide the users state and bound action handlers as props
-
test/users-fetch-logic.spec.js - testing usersFetch logic in isolation
npm install # install dependencies
npm start # builds and runs dev server
Click fetch button which dispatches a simple USERS_FETCH
action, that the logicMiddleware picks up, hands to fetchUsersLogic and runs the code in the process hook creating async fetch which eventually resolves and is handed to usersFetchFullfilled
action creator or usersFetchRejected
error action creator before being dispatched.
Note: To slow things down so you can interactively cancel and test the take latest functionality, I have added a delay query param to the URL which delays the response by 4 seconds. This allows you time to click cancel or to click fetch multiple times to see that it will only result in the latest result being fulfilled.