@oqton/redux-black-box
v1.1.5
Published
Declare side effects as black boxes in redux: an alternative for redux-thunk, redux-saga, redux-loop, ...
Downloads
541
Readme
Redux-black-box
Redux is a powerful way of thinking:
- You gather the complete state of your system in one place
- You explicitly describe how actions can transform the state of the system
The advantage of this is that such an implementation is easy to reason about, verify, or even prove to be correct.
However, a redux system often has to interact with other systems that we can not or do not want to describe as a redux system itself. E.g. a timer, a remote call, a web socket, ...
Other libraries like redux-thunk, redux-saga or redux-observable argue: "This can not be described as a state machine (e.g. because of the side effects), so it does not fit in redux", and solve this by adding an extra, parallel system that stands next to it. By partially bypassing redux like this we lose a lot of its advantages and we add a lot of complexity.
- The state is no longer contained in the redux store alone. These parallel systems have their own state that is not captured; most importantly, which asynchronous interactions exist. This makes it more complex to cancel in flight asynchronous interactions, and to know the state in which the combined system will be left after cancellation.
- The logic is no longer contained in the reducer alone. This makes it much harder to reason about the application. Moreover, the interaction between the redux system and the parallel system is complex.
The goal of this library is to solve these asynchronous interactions with other systems WITHIN the redux frame of thought. We do this by separating the declaration of the interaction, which we call a black box, from its execution. The declaration of the black box is done in the reducer while the execution is done by the middleware.
How to use black boxes in redux
Declaring a black box is simply done by adding an instance of a class that extends AbstractBlackBox
to the redux state.
Note that declaring the black box is side-effect-free.
The custom middleware will take care of the life cycle and execution of the black box.
import { createStore, applyMiddleware } from 'redux';
import { blackBoxMiddleware } from '@oqton/redux-black-box';
const reducer = (state, action) => {
switch(action.type) {
case "FETCH_REQUEST":
return {
...state,
call: new PromiseBlackBox( // predefined class extending AbstractBlackBox
() => fetch('http://www.server.org') // asynchronous side effect
.then(res => res.json()) // decode the JSON response.
.then(res => ({type:"FETCH_SUCCESS", res})) // return action with results
)
};
case "FETCH_SUCCESS":
return {
...state,
result: action.res,
call: null
};
case "FETCH_ABORT":
return {
...state,
call: null
};
default:
return state;
}
}
const store = createStore(reducer, undefined, applyMiddleware(blackBoxMiddleware));
This example implements a remote call using a PromiseBlackBox
. The action returned by the promise is automatically dispatched to the redux store.
Note that this is only one of many available types of black box.
Redux-black-box rules
Let us have a look at the three principles of redux and derive the rules for black boxes.
Single source of truth
The state of your whole application is stored in an object tree within a single store.
Black boxes should be declared as part of the redux state
The state of the whole application including the existence of the asynchronous interactions is captured in a single redux store. The internal state of the black boxes, however, is not defined in the redux store. These bits have to be explicitly declared and are clearly contained.
The life cycle of the asynchronous interaction described by a black box is linked to its existence in the redux state. When the declaration of a black box is removed from the redux state, the middleware will prevent it from further interacting with it and will, if possible, stop the execution of the asynchronous code. (E.g. in case of a cancellable promise.)
E.g. using a black box we declare that a fetch call should happen, but we do not describe the state of the fetch call (uploading, waiting for response, downloading, done, ...) in redux.
State is read-only
The only way to change the state is to emit an action, an object describing what happened.
Black boxes do not expose their internal state, but they can communicate it using actions
A black box, hence the name, does not expose any state that could be considered as not read-only. It can only communicate with the redux store using actions and can only do this as long as it is part of the redux state.
E.g. we cannot get any information about the fetch call that it does not communicate using an action nor does it have methods that we can use to affect the fetch call.
Changes are made with pure functions
To specify how the state tree is transformed by actions, you write pure reducers.
Black boxes are declared in the reducer, but executed by the middleware
The reducer function that transforms the state tree is thus a pure, synchronous function.
E.g. the declaration of a fetch black box in the reducer does not trigger the fetch, but the middleware observing the redux state change will.
Example use case of redux-black-box: Data loader
Read about the implementation of a complex data loader in Example to better understand how to use redux-black-box and experience its benefits.
Predefined types of black boxes
In most cases, you will not have to define a black box from scratch.
Instead, you use one of a number of predefined types, such as the PromiseBlackBox
.
These are described here.
Comparison against alternative libraries
We compare redux-black-box with several state of the art libraries in this document. We cover redux-thunk, redux-saga, redux-observable and redux-loop.
Frequently asked questions
Can be found at docs/FAQ.md.
Design decisions
Documentation about design decisions can be found here. This is useful to read for contributors and may also be interesting if you want to implement your own custom black box.
Questions and bugs
Bugs and pull requests can be submitted to GitHub.
Other questions should be posted to Stack Overflow using the [redux-black-box]
tag.
We will attempt to answer your questions there.