Handy if you want to stash data, then systematically process it later. If you hook it up with redux and save your state, you've got yourself an offline persistence queue.
This module is an internally redux powered queue for the storage and deferred processing of objects of multiple named types by a specified number of parallel workers.
It coordinates the on-demand sequential application of per-type custom handler functions to the stored objects.
Can be used as a standalone module without having to touch redux, alternatively use middleware to join to an external redux store.
Suited to complex deferred offline persistence handling, eg with validation, saving and linking stages.
Let's save our dog, Buster. Grab the module from npm with yarn
or npm
:
yarn add redux-worker-queue
npm install --save redux-worker-queue
Import the WorkerQueue
constructor, and initialize the queue with a TypeRegistration
. We'll include a type
of 'PET', and pet handlers which return promises.
import WorkerQueue from 'redux-worker-queue';
// Initialise queue
const petType = {
type: 'PET',
handlers: [
isMyPetValidAsync,
saveMyPetAsync,
]
}
const myPetsQueue = new WorkerQueue(petType);
myPetsQueue.init();
Call addOrUpdateQueueItem()
with a NewQueueItem
- specifying the type
as 'PET', and including Buster's custom information as the payload
.
const myPet = {
name: 'Buster',
panting: true,
};
// Add Buster to the queue.
const id = myQueue.addOrUpdateQueueItem({
type: 'PET',
payload: myPet,
});
Buster's QueueItem
is now stored in the queue.
When you're ready, call flush()
.
myPetsQueue.flush();
Buster will now be handed to isMyPetValidAsync
...and if he's valid, to saveMyPetAsync
...and if that's successful, he's removed off the queue. Voilà! You've saved him!
The handlers provided to the queue for each type are applied to each QueueItem
of their specific type, in order of their registration. Each handler can block progression to the next.
The handler functions must be pure functions which take a QueueItem
as their first argument, and return a promise. The promise must always resolve as below:
{
ok: Boolean,
item: QueueItem
}
There are three different outcomes from the handler response: OK, halted, locked.
Triggered by handler resolving to:
`{ ok: true, item: possibly changed QueueItem }`
If the QueueItem
is now ready for the next handler, set ok: true
. A queue item is only processed by the next handler once the preceding handler has resolved to ok: true
.
Once the last handler resolves in this way, the item is removed from the queue...Christmas!
Triggered by handler resolving to:
`{ ok: false, item: a changed QueueItem }`
A halt stops the subsequent handlers being called until the QueueItem
is updated and flush()
called again.
For example, a validation handler that does validation would probably put its validation errors into QueueItem.errors
to be shown to a user for action, and return the updated QueueItem
along with ok: false
. The user action should trigger an update to the item by passing the changed QueueItem
to addOrUpdateQueueItem()
. The update to the QueueItem
will release the halt.
Triggered by a rejected handler promise, a handler throwing an error, or a handler resolving to:
`{ ok: false, item: an unchanged QueueItem }`
A QueueItem
will be locked out of processing temporarily if
ok: false
is resolved from the handler promise and no change was made to theQueueItem
, or- a handler promise is rejected
- a handler throws an error
The handler promise should always resolve.
If a handler rejects or throws an error, the error is logged to the console if possible to help you.
Once the lockoutPeriod
has elapsed (as specified on WorkerQueue
initialisation or set dynamically), the QueueItem
will be released for processing again.
One or more workers process the queue, applying the correct handler for the QueueItem. The number of concurrent workers can be set in the initialization options or with WorkerQueue.workers
.
Import the queue middleware, and add the queue reducers:
import { createStore, applyMiddleware } from 'redux'
import WorkerQueue from 'redux-worker-queue';
import myAwesomeReducer from './reducers';
import {
isMyPetValidAsync,
saveMyPetAsync,
linkMyPetsAsync
} from './handlers';
// Initialise the Worker WorkerQueue as usual, but before using middleware
const workerQueue = new WorkerQueue({
type: 'PET',
handlers: [
isMyPetValidAsync,
saveMyPetAsync,
linkMyPetsAsync
]
});
// Create your store with a workerQueue key and workerQueue middleware applied
let store = createStore(
{
myState: myAwesomeReducer,
workerQueue: workerQueue.reducer
},
applyMiddleware(workerQueue.middleware)
);
// Don't call workerQueue.init().
Use the queue as before, but this time you can also send actions directly:
// We want to save Buster!
const myPet = {
name: 'Buster',
panting: true,
};
// Add Buster to the queue with a redux action
const addBusterAction = myQueue.actions.addOrUpdateItem({
type: 'PET',
payload: myPet,
});
store.dispatch(addBusterAction); // Done!
// Or, use the queue as directly
const id = myQueue.addOrUpdateQueueItem({
type: 'PET',
payload: myPet,
}); // Done as well.
If you want to use a custom root key, specify it with the reduxRootSelector
setting:
// Initialise the WorkerQueue as usual, but passing reduxRootSelector setting
const workerQueue = new WorkerQueue({
type: 'PET',
handlers: [
isMyPetValidAsync,
saveMyPetAsync,
linkMyPetsAsync
]
}, {
// Add reduxRootSelector to our new root key
reduxRootSelector: state => state.myAwesomeQueue
});
// Create your store the custom queue key
let store = createStore(
{
myState: myAwesomeReducer,
// Same key as specified in reduxRootSelector
myAwesomeQueue: workerQueue.reducer,
},
applyMiddleware(workerQueue.middleware)
);
Call new WorkerQueue()
to obtain the queue instance. Allows only one instance to be made.
Takes a single TypeRegistration
or array of TypeRegistration
objects, and an optional Settings
object.
const myQueue = new WorkerQueue(TypeRegistraton|[TypeRegistration],Settings?);
Initialises the queue. Internally, redux is used, and this sets up the store. Call immediately after obtaining the queue instance and before adding any items to the queue. Omit if using external redux integration.
Takes a single TypeRegistration
object to register a type of QueueItem
, and handlers for that type. Use if you need to add a QueueItem
type after initialising the queue.
myQueue.registerQueueItemType(TypeRegistration)
Called to add a new item to the queue, or update an existing one. Returns the ClientMutationId
of the QueueItem
.
myQueue.addOrUpdateQueueItem(
item: QueueItem | NewQueueItem
)
Take the ClientMutationId
of a QueueItem
. Returns a promise, resolving to a copy of QueueItem
from the queue, or undefined
if not found.
myQueue.getItem(
id: ClientMutationId
)
Returns the handlers for the specified QueueItem
type.
myQueue.getHandlersForType(type: String)
Called to remove QueueItem
from the queue, as identified by its ClientMutationId
.
myQueue.removeItem(clientMutationId: string)
Danger! Wipes the queue.
myQueue.clearQueue()
Returns the ordering settings of the queue.
Set this to the number of workers required.
Set this to the number of milliseconds for which a QueueItem
should be locked out.
{
type: string,
handlers: [
(item: QueueItem): Promise<{ ok: boolean, item: QueueItem }>
]
}
{
workers?: number,
reduxRootSelector?: globalState => WorkerQueueState,
lockoutPeriod?: Integer, >= 0, in milliseconds
}
Can be used to make a brand new QueueItem
. If not set, clientMutationId
is automatically generated.
NewQueueItem {
type: String;
payload: object;
clientMutationId?: string|number;
}
A QueueItem
is guaranteed to have all these properties. The createdAt
property cannot be overridden. The clientMutationId
is the unique identifier.
QueueItem {
type: string;
payload: object;
errors: array;
clientMutationId: ClientMutationId;
createdAt: ISO-8601 date string;
}
All QueueItem
objects on the queue are identified with a clientMutationId
property, which is autogenerated if not provided on creation. It's a number or alphanumeric string.
ClientMutationId: string|number