@talend/react-cmf-cqrs
Version:
@talend/react-cmf plugin for CQRS backend architecture
188 lines (141 loc) • 5.95 kB
Markdown
# Content Management Framework for CQRS (aka cmf-cqrs)
This is a library to help you to build configurable React App with CQRS pattern.
[![Travis CI][travis-ci-image] ][travis-ci-url]
[![NPM][npm-icon] ][npm-url]
[![dependencies][dependencies-image] ][dependencies-url]
[![devdependencies][devdependencies-image] ][devdependencies-url]
[npm-icon]: https://img.shields.io/npm/v/@talend/react-cmf-cqrs.svg
[npm-url]: https://npmjs.org/package/@talend/react-cmf-cqrs
[travis-ci-image]: https://travis-ci.org/Talend/ui.svg?branch=master
[travis-ci-url]: https://travis-ci.org/Talend/ui
[dependencies-image]: https://david-dm.org/Talend/ui/status.svg?path=packages/cmf-cqrs
[dependencies-url]: https://david-dm.org/Talend/ui?path=packages/cmf-cqrs
[devdependencies-image]: https://david-dm.org/Talend/ui/dev-status.svg?path=packages/cmf-cqrs
[devdependencies-url]: https://david-dm.org/Talend/ui?path=packages/cmf-cqrs&type=dev
## Content
This package provides tools to deal with cqrs backend allowing websocket handling :
- acknowledgement actions
- ACKDispatcher component
- Smart Websocket middleware
- ACK reducer
## How it works
- to start the websocket with smartWebsocket middleware :
```javascript
ws = new SmartWebsocket(urlPrefix, {
onOpen: () => dispatch({ type: SOCKET_ON_OPEN }),
onClose: () => dispatch({ type: SOCKET_ON_CLOSE }),
onMessage: messageEvent => {
socketListener.forEach(func => func(messageEvent, dispatch, getState, ws));
},
onError: errorEvent => {
dispatch({ type: SOCKET_ON_CLOSE, event: errorEvent });
},
onPing: event => {
ws.pingTimeoutId = event.pingTimeoutId;
},
onPingTimeout: () => {
dispatch({ type: SOCKET_ON_PING_TIMEOUT });
},
...socketOptions,
});
```
socketOptions is optionnal but allows websocket configuration from middleware instanciation within your application.
```javascript
{
checkInterval: 5000,
pingInterval: 10000,
pingTimeoutDelay: SOCKET_ON_PING_TIMEOUT_DELAY,
}
```
- checkInterval : max duration between 2 websocket connections trials if closed
- pingInterval : duration between ping message from the webapp to the server, like a heartbeat of the connection
- pingTimeoutDelay : duration after which a PING message not being answered by a PONG will trigger a SOCKET_ON_PING_TIMEOUT and force close of the current connection
In onMessage event, you should get middleware handlers as well.
- On the reducer, actions handled :
- ACK_ADD_CONTEXT : Used to add a new request on stack
- ACK_RECEIVE_MESSAGE : Used when a message come from the ws
- ACK_DELETE : Used when you want to delete a handler
## SocketMiddleware
This middleware serve two purpose
listening to action dispatched to the reducer, and be able to send message trought the websocket
listening to message comming from the websocket and dispatching message to the reducer
### Configuration
```javascript
import cmf from '@talend/react-cmf';
import cqrsModule, { middlewares as cqrsMiddlewares } from '@talend/react-cmf-cqrs';
const websocketMiddleware = cqrsMiddlewares.createWebsocketMiddleware(
API['cqrs-websocket'],
[...actionListeners],
[...socketListener],
{ ...socketOptions },
);
cmf.bootstrap({
appId: 'app',
// ...
modules: [cqrsModule]
middlewares: [websocketMiddleware]
});
```
the action listener recieve for each action
the store state ,before the action is applied
the action dispatched to the store reducer
the new state resulting from the action to be applied
the value returned by the actionListener get sent trought the websocket.
```javascript
/** an example of an action listener sending all action trought the ws **/
function myActionListener(previousState, action, nextState) {
return JSON.stringify(action);
}
```
the socketListener recieve for each message
the messageEvent containing all information about the message recieved
the store.dispatch function so you can dispatch an action
the store.getState function in case you want to check the store state before doing anything
the smartWebSocket itself so the listener can access to the websocket itself (eg: datastreams pong socket listener that get timeoutId and clear it before it's execution)
```javascript
/** an example of an message listener dispatching each message recieved **/
function myMessageListener(messageEvent, dispatch, getState, smartWebSocket) {
dispatch({type: 'MESSAGE RECIEVED', message: JSON.parse(messageEvent.data);})
}
```
Additionnaly the websocket middleware dispatch action on socket state change and on message being recieved.
on socket connection open success the following action get dispatched
```
{ type: SOCKET_ON_OPEN }
```
on socket connection being closed the following action get dispatched
```
{ type: SOCKET_ON_CLOSE }
```
on socket connection being closed by an error the following action get dispatched
```
{ type: SOCKET_ON_ERROR, event: errorEvent }
```
on socket connection receiving a message, no action get dispatched but socketListeners are called which will take care of dispactching action or not or even do something else
on socket connection timeout reached, the following action get dispatched
```
{ type: SOCKET_ON_PING_TIMEOUT }
```
## Hook
Once boostraped with the `createWebsocketMiddleware`, you can use the hook to get the websocket instance
```javascript
import { useWebSocket } from '@talend/react-cmf-cqrs';
function MyComponent() {
const { lastJsonMessage, lastMessage, readyState, sendMessage, sendJsonMessage } = useWebsocket();
useEffect(() => {
makeStuff(lastJsonMessage);
}, [lastJsonMessage]);
}
```
We can provide a list of ignored events (because we make some pong request to maintain the connection) this way :
```javascript
import { useWebSocket } from '@talend/react-cmf-cqrs';
function MyComponent() {
const { lastJsonMessage, lastMessage, readyState, sendMessage, sendJsonMessage } = useWebsocket([
'pong',
]);
useEffect(() => {
makeStuff(lastJsonMessage);
}, [lastJsonMessage]);
}
```