Separating async business logic from user interface in React applications. Especially in those that use Redux.
The react-redux-cases
library helps you extract plain business logic (i.e. cases) into independent classes. Library hooks connect these cases to React components. So it prevents mixing async fetching, state management, UI styling in a component. Instead, your components stay clear and simple.
$ npm install react-redux-cases
Requires react-redux.
In terms of the react-redux-cases
library, the case is a separate unit that covers one application feature. We might also call it an application service or a use case.
The case is implemented as a class with interface:
interface Case<Res, Err, P> {
execute(runParams: P): Promise<Result<Res, Err>>;
onAbort?: () => void;
}
Rule: the execute
method must not throw an exception. Instead, it returns the Promise
of the Result
object.
Example of Case
:
import { Case } from 'react-redux-cases';
import { AppDispatch, AppGetState, updateList } from './my/app/redux';
import { Todo, apiGetTodoList } from './my/app/todo';
class LoadTodoListCase implements Case {
private dispatch: AppDispatch;
private getState: AppGetState;
// inject all dependencies in constructor
constructor(dispatch: AppDispatch, getState: AppGetState) {
this.dispatch = dispatch;
this.getState = getState;
}
// static factory method creates the LoadTodoListCase instance
static create(dispatch: AppDispatch, getState: AppGetState) {
return new LoadTodoListCase(dispatch, getState);
}
// use case implementation
async execute(todoFilter: string) {
// get state from redux store
const userId = this.getState().user.id;
// call async API
const result = await apiGetTodoList({ user: userId, filter: todoFilter });
if (result.isErr()) {
// manage error
console.log('SynchronizeTodoListCase:', result.error);
// return failed result
return result;
}
// manage successful result
// update redux store
this.dispatch(updateList(result.value.data));
// return successful result
return result;
}
}
The core of this case is the mandatory function execute
. As we can see, it implements the following scenario:
- get the string
todoFilter
from the function parameter - get the ID of the user from the redux store
- call the async API service
/list
to get filtered To-Do list data from backend server - on failure, log an error message
- if successful, update redux store with new To-Do list
- return
Result
object with To-Do list data
This case is separated from the rest of the application. It declares all its dependencies in the contructor
, making it well testable. Our case requires redux getState
and dispatch
methods.
Static factory method create
is not necessary but it is very useful.
Rule: Factory method must create an instance and not throw an exception.
The execute
method must not throw an exception. Instead, it must return a Result
object, which is a union type of a success or error value.
type Result<V, E> = Ok<V> | Err<E>;
The Ok
object wraps a value
and offers it via the result.value
getter.
The Err
object wraps an error
and offers it via the result.error
getter.
Both Ok
and Err
objects implement isOk()
and isErr()
methods, which act as type guards.
Examples of creating a Result
instance:
import { ok } from 'react-redux-cases';
// Ok result
const okResult = ok('success');
if (okResult.isOk()) {
console.log(okResult.value); // -> 'success'
}
if (okResult.isErr()) {
// -> never
}
import { err } from 'react-redux-cases';
// Err result
const errResult = err('error message');
if (errResult.isErr()) {
console.log(errResult.error); // -> 'error message'
}
if (errResult.isOk()) {
// -> never
}
Example of an API function that returns a Result
:
/**
* `apiGetTodoList` function returns the `Result` object
*/
async function apiGetTodoList(
params: { user: string; filter: string },
abortController?: AbortController,
) {
try {
const response: Awaited<AxiosResponse<Todo[]>> = await axios({
method: 'get',
url: 'list',
params,
signal: abortController.signal,
});
return ok(response.data);
} catch (e) {
return err(e);
}
}
/**
* usage the `apiGetTodoList` function in `execute` method
*/
class LoadTodoListCase implements Case {
// ...
async execute(todoFilter: string) {
// ...
// Result object
const result = await apiGetTodoList({ user: userId, filter: todoFilter });
if (result.isErr()) {
// result is Err object
// do something with result.error
return result;
}
// result is Ok object
// do something with result.value
return result;
}
}
No exception, just a simple object.
Cases are independent pieces of code. How can we use them in React components?
As an adapter, we can choose from prepared library hooks useCase
, useCaseState
, useReduxCase
, useReduxCaseState
.
Each of the hooks gets a Case factory method as a parameter. Factory method must create an instance and not throw an exception.
The useReduxCaseState
and useReduxCase
hooks provide the Redux getState
and dispatch
methods as parameters for the factory methods.
Examples of factory methods:
// 1. example - using the static factory method
const case1 = useReduxCaseState(LoadTodoListCase.create);
// 2. example - this is the equivalent expression, using a case constructor
const case2 = useReduxCaseState((dispatch, getState) => new LoadTodoListCase(dispatch, getState));
// 3. example - injecting an additional dependency into case
const additionalDependency = useSomething();
const case3 = useReduxCaseState((dispatch, getState) =>
LoadTodoListCase.create(dispatch, getState, additionalDependency),
);
Each of four hooks creates a run
function. React component then can call this run
function to execute the case.
The run
function instantiates the Case
object via its factory method, passes it all dependencies, calls the execute
function with arguments passed to run
function, and finally returns Result
object.
Moreover useCaseState
and useReduxCaseState
returns state
object, so that the component can watch the async process state.
Usage in component:
import { useReduxCaseState } from 'react-redux-cases';
import { LoadTodoListCase } from './my/app/todo';
// list data comes from redux store
const FilteredTodoList = ({ list }: { list: Todo[] }) => {
// make connection with our LoadTodoListCase
const { run, error, state } = useReduxCaseState(LoadTodoListCase.create);
const handleChangeFilter = (newFilter: string) => {
// run the execute(newFilter) method of the LoadTodoListCase
run(newFilter);
};
return (
<>
<Filter onChange={handleChangeFilter} />
{state.isPending && <Spinner />}
{!state.isPending && state.isRejected && <ErrorPanel>{String(error)}</ErrorPanel>}
<List list={list} />
</>
);
};
Hook | Case Factory | Async State Monitoring |
---|---|---|
useReduxCaseState(caseFactory) |
(dispatch, getState) => Case |
Yes |
useReduxCase(caseFactory) |
(dispatch, getState) => Case |
No |
useCaseState(caseFactory) |
() => Case |
Yes |
useCase(caseFactory) |
() => Case |
No |
Besides, each hook returns functions:
run: async (runParams) => Promise<Result>
abort: () => void
Cases may call other cases within the execute
method. Components call such a compound case once and does not need to trigger a chain of cases using the useEffect
hook.
Example:
class AddTodoItemCase implements Case {
// ...
async execute(todoItem: Todo) {
// call API
const result = await apiAddTodoItem({ item: todoItem });
if (result.isErr()) {
// result is Err object
// do something with result.error
return result;
}
// New item is created on backend,
// so we need to update the todo list.
// Create the LoadTodoListCase:
const loadCase = LoadTodoListCase.create(this.dispatch, this.getState);
// and execute it:
const loadingResult = await loadCase.execute('');
if (loadingResult.isErr()) {
return loadingResult;
}
return result;
}
}
The Case
interface offers onAbort
method. When the component is unmounted, the onAbort
method is callled. It is up to you how your case will behave in this situation. A common approach is to use AbortController API.
It is also possible to abort the case manually. All four hooks useCase
, useCaseState
, useReduxCase
, useReduxCaseState
provide an abort
method that can be called in components.
Aborted case does not change any of the value
, error
, state
values returned from the useCaseState
or useReduxCaseState
hook. E.g. manually aborted pending case remains pending. Therefore, the last properly finished case will return the correct value
, error
and state
.
Example of the LoadTodoListCase
with AbortController
:
class LoadTodoListCase implements Case {
private dispatch: AppDispatch;
private getState: AppGetState;
private abortController?: AbortController;
constructor(dispatch: AppDispatch, getState: AppGetState, abortController?: AbortController) {
this.dispatch = dispatch;
this.getState = getState;
this.abortController = abortController;
}
static create(dispatch: AppDispatch, getState: AppGetState) {
return new LoadTodoListCase(dispatch, getState, new AbortController());
}
async execute(todoFilter: string) {
// ...
// pass the AbortController signal to API
const result = await apiGetTodoList(
{ user: userId, filter: todoFilter },
this.abortController?.signal,
);
if (result.isErr()) {
// aborted API request returns an error
console.log('apiGetTodoList error', result.error);
// the case ends, so redux is not updated
return result;
}
// update redux state
this.dispatch(updateList(result.value.data));
return result;
}
onAbort() {
this.abortController?.abort();
}
}
When we type a few characters in the filter input field, a series of request is sent. To prevent a request race, we need to abort old requests every time a new character is typed.
Example of updated FilteredTodoList
component:
const FilteredTodoList = ({ list }: { list: Todo[] }) => {
const { run, error, state, abort } = useLoadTodoList();
const handleChangeFilter = (newFilter: string) => {
// abort previous requests
abort();
// make new request
run(newFilter);
};
// ...
};
You can explore and try the sample application.
The useReduxCaseState(caseFactory)
hook returns run
and abort
methods and values for state monitoring. Passes the Redux dispatch
and getState
methods to caseFactory
as arguments.
Parameters
caseFactory
:(dispatch, getState) => Case
- it must not throw an exception. The returned object should implement theCase
interface.
Returns
Case controlling:
run
:async (runParams) => Promise<Result>
- use therun(runParams)
method to invoke theCase
execute(runParams)
methodabort
:() => void
- calling theabort()
method invokes theCase
onAbort()
method
Async state monitoring:
value
: resolved promise value from therun
method, it is unwrappedvalue
of theResult
objecterror
: rejected promise value from therun
method, it is unwrappederror
of theResult
objectstate
: state objectstate
: 'initial' | 'pending' | 'resolved' | 'rejected'isInitial
: boolean - true when norun
has startedisPending
: boolean - true whenrun
method is awaitingisResolved
: boolean - true whenrun
was resolvedisRejected
: boolean - true whenrun
was rejectedisFinished
: boolean - true whenrun
was resolved or rejected
actions
: control the state manually (rarely usable)start
:() => void
- marks the state as 'pending'resolve
:(value) => void
- marks the state as 'resolved' and sets the resolvedvalue
reject
:(error) => void
- marks the state as 'rejected' and sets the rejectederror
valuereset
:() => void
- marks the state as 'initial' and resetsvalue
anderror
The useReduxCase(caseFactory)
hook returns run
and abort
methods. Passes the Redux dispatch
and getState
methods to caseFactory
as arguments.
This is similar to
useReduxCaseState
, but without state monitoring.
Parameters
caseFactory
:(dispatch, getState) => Case
Returns
run
:async (runParams) => Promise<Result>
abort
:() => void
The useCaseState(caseFactory)
hook returns run
and abort
methods and values for state monitoring.
This is similar to
useReduxCaseState
, but without providingdispatch
andgetState
methods forcaseFactory
.
Parameters
caseFactory
:() => Case
Returns
run
:async (runParams) => Promise<Result>
abort
:() => void
value
error
state
:{state, isInitial, isPending, isResolved, isRejected, isFinished}
actions
:{start, resolve, reject, reset}
The useCase(caseFactory)
hook returns run
and abort
methods.
This is similar to
useReduxCaseState
, but without providingdispatch
andgetState
methods forcaseFactory
and without state monitoring.
Parameters
caseFactory
:() => Case
Returns
run
:async (runParams) => Promise<Result>
abort
:() => void
The Case
is interface.
Methods
execute
:async (runParams) => Result
- async function returns theResult
object. It must not throw an exception. Therun
method of the hooks calls theexecute
method of the case.onAbort
:() => void
- method is optional. Theabort
method of the hooks calls theonAbort
method of the case.
Example:
import { Case, ok } from 'react-redux-cases';
class MyCase implements Case {
constructor(readonly dispatch: AppDispatch, readonly getState: AppGetState) {}
// static factory method
static create(dispatch: AppDispatch, getState: AppGetState) {
return new MyCase(dispatch, getState);
}
// use case implementation
async execute(param: string) {
// ...
// return Result
return ok(someResult);
}
}
Result
is a union type of the Ok
or Err
value.
type Result<V, E> = Ok<V> | Err<E>;
Class Ok
wraps a value
of any type. To create a new instance, you can use the constructor or helper function ok(value)
.
Example with constructor:
import { Ok } from 'react-redux-cases';
const result = new Ok({ title: 'Success' });
Example with ok(value)
function:
import { ok } from 'react-redux-cases';
const result = ok({ title: 'Success' });
Class members
constructor(value)
- thevalue
can be of any typevalue
: readonly valueisOk()
: type guard, returns trueisErr()
: type guard, returns false
Class Err
wraps an error
of any type. To create a new instance, you can use the constructor or helper function err(error)
.
Example with constructor:
import { Err } from 'react-redux-cases';
const result = new Err({ reason: 'Bad credentials' });
Example with err(error)
function:
import { err } from 'react-redux-cases';
const result = err({ reason: 'Bad credentials' });
Class members
constructor(error)
- theerror
can be of any typeerror
: readonly error valueisOk()
: type guard, returns falseisErr()
: type guard, returns true
The ok(value)
helper function creates a new instance of the Ok
class.
ok
:(value) => Ok
The err(error)
helper function creates a new instance of the Err
class.
err
:(error) => Err
useAsyncState()
helps monitor the state of an async process. Hook stores the result value or error of an async process and its current state. It does not control the process itself.
Returns
value
: resolved valueerror
: rejected valuestate
: the state of the async processstate
: 'initial' | 'pending' | 'resolved' | 'rejected'isInitial
: boolean - true when state is 'initial'isPending
: boolean - true when state is 'pending'isResolved
: boolean - true when state was 'resolved'isRejected
: boolean - true when state was 'rejected'isFinished
: boolean - true when state was 'resolved' or 'rejected'
actions
: setting the state and resultstart
:() => void
- marks the state as 'pending'resolve
:(value) => void
- marks the state as 'resolved' and sets the resolvedvalue
reject
:(error) => void
- marks the state as 'rejected' and sets the rejectederror
valuereset
:() => void
- marks the state as 'initial' and resetsvalue
anderror
toundefined
Although the purpose of this library has remained the same, it is not backward compatible with version 0.x. With care and appropriate effort, you can rewrite v0 hooks for object cases (useObjReduxCaseState
, useObjReduxCase
, useObjCaseState
, useObjCase
) with v1 hooks and cases. Hooks for functional cases are removed, so a refactoring to v1 object cases is necessary.
MIT