A library for canceling asynchronous requests that combines the Saborter library and React.
The documentation is divided into several sections:
npm install saborter @saborter/react
# or
yarn add saborter @saborter/react- The
aborterfield always has the same reference to theAborterinstance. - Automatically abort the request when the component is unmounted.
- Automatically unsubscribe from all listeners when the component is unmounted.
import { useAborter } from '@saborter/react';
const Component = () => {
// Create an Aborter instance via the hook
const { aborter } = useAborter();
// Use for the request
const fetchData = async () => {
try {
const data = await aborter.try((signal) => fetch('/api/data', { signal }));
console.log('Data received:', data);
} catch (error) {
console.error('Request error:', error);
}
};
};const { aborter } = useAborter(props?: UseAborterProps);| Parameter | Type | Description | Required |
|---|---|---|---|
props |
UseAborterProps |
Aborter configuration options | No |
UseAborterProps:
{
/**
Callback function for abort events.
Associated with EventListener.onabort.
It can be overridden via `aborter.listeners.onabort`
*/
onAbort?: OnAbortCallback;
/**
A function called when the request state changes.
It takes the new state as an argument.
Can be overridden via `aborter.listeners.state.onstatechange`
*/
onStateChange?: OnStateChangeCallback;
/**
A flag responsible for releasing resources.
This includes unsubscribing, clearing fields, and removing references to passed callback functions.
@default true
*/
dispose?: boolean;
}aborter: Aborter
Returns the Aborter instance.
const { aborter } = useAborter();
// Using signal in the request
fetch('/api/data', {
signal: aborter.signal
});requestState: RequestState
The current value of the request's state. May be undefined if the state has not yet been set.
The field is a react state associated with the aborter.listeners.state.value field.
const { requestState } = useAborter();
console.log(requestState); // 'cancelled' / 'pending' / 'fulfilled' / 'rejected' / 'aborted'// The type can be found in `saborter/types`
const reusableAborter = useReusableAborter(props?: ReusableAborterProps);| Parameter | Type | Description | Required |
|---|---|---|---|
props |
ReusableAborterProps |
ReusableAborter configuration options | No |
ReusableAborterProps:
{
/**
* Determines which listeners are carried over when the abort signal is reset.
* - If `true`, all listeners (both `onabort` and event listeners) are preserved.
* - If `false`, no listeners are preserved.
* - If an object, specific listener types can be enabled/disabled individually.
*/
attractListeners?: boolean | AttractListeners;
}signal: AbortSignal
Returns the AbortSignal associated with the current controller.
const reusableAborter = useReusableAborter();
// Using signal in the request
fetch('/api/data', {
signal: reusableAborter.signal
});abort(reason?): void
Parameters:
reason?: any- the reason for aborting the request.
Immediately cancels the currently executing request.
Note
Can be called multiple times. Each call will restore the signal, and the aborted property will always be false.
import { useState } from 'react';
import { AbortError } from 'saborter/errors';
import { useAborter } from '@saborter/react';
const Component = () => {
// Create an Aborter instance via the hook
const { aborter } = useAborter();
const [user, setUser] = useState(null);
const [loading, setLoading] = useState(false);
// Use for the request
const fetchData = async () => {
try {
setLoading(true);
const user = await aborter.try((signal) => fetch('/api/user', { signal }));
setUser(user);
} catch (error) {
if (error instanceof AbortError) {
// An abort error will occur either when the `aborter.abort()` method is called
// or when the component is unmounted.
console.error('Abort error:', error);
}
console.error('Request error:', error);
} finally {
setLoading(false);
}
};
return <h1>{loading ? 'Loading...' : user.fullname}</h1>;
};import { useState } from 'react';
import { AbortError } from 'saborter/errors';
import { useAborter } from '@saborter/react';
const Component = () => {
// Create an Aborter instance via the hook
const { aborter, loading } = useAborter();
const [user, setUser] = useState(null);
// Use for the request
const fetchData = async () => {
try {
const user = await aborter.try((signal) => fetch('/api/user', { signal }));
setUser(user);
} catch (error) {
if (error instanceof AbortError) {
// An abort error will occur either when the `aborter.abort()` method is called
// or when the component is unmounted.
console.error('Abort error:', error);
}
console.error('Request error:', error);
}
};
return <h1>{loading ? 'Loading...' : user.fullname}</h1>;
};import { AbortError } from 'saborter/errors';
import { useAborter } from '@saborter/react';
const Component = () => {
const { aborter } = useAborter({
onAbort: (error) => {
if (error.type === 'aborted' && error.initiator === 'component-unmounted') {
console.log('Component is unmounted!');
}
}
});
const fetchData = async () => {
const user = await aborter.try((signal) => fetch('/api/user', { signal }));
};
};If you have an aborter instance that was created behind a component, for example, in a parent component, but you want to abort the request when the child is unmounted, you can use the useAbortWhenUnmount hook.
import { useAborter, useAbortWhenUnmount } from '@saborter/react';
const Child = ({ aborter }) => {
useAbortWhenUnmount(aborter);
return <div>Child component</div>;
}
const Parent = () => {
const { aborter } = useAborter();
// Use for the request
const fetchData = async () => {
try {
const user = await aborter.try((signal) => fetch('/api/user', { signal }));
setUser(user);
} catch (error) {
if (error instanceof AbortError && error.initiator === 'component-unmounted') {
// handling request interruption due to component unmounting
}
}
};
return (
<div>
Parent Component
<Child aborter={aborter}>
</div>
);
};import { useEffect } from 'react';
import { useReusableAborter } from '@saborter/react';
const Component = () => {
const aborter = useReusableAborter();
useEffect(() => {
// Attach listeners
aborter.signal.addEventListener('abort', () => console.log('Listener 1'));
aborter.signal.addEventListener('abort', () => console.log('Listener 2'), { once: true }); // won't be recovered
// Set onabort handler
aborter.signal.onabort = () => console.log('Onabort handler');
}, []);
const handleFirstClick = () => {
// First abort
aborter.abort('First reason');
// Output:
// Listener 1
// Listener 2 (once)
// Onabort handler
// The signal is now a fresh one, but the nonβonce listeners and onabort are reattached
aborter.signal.addEventListener('abort', () => console.log('Listener 3')); // new listener, will survive next abort
};
const handleSecondClick = () => {
// Second abort
aborter.abort('Second reason');
// Output:
// Listener 1
// Onabort handler
// Listener 3
};
return (
<div>
<button onClick={handleFirstClick}>First abort</button>
<button onClick={handleSecondClick}>Second abort</button>
</div>
);
};MIT License - see LICENSE for details.
