2019-12-04 12:45:49 +01:00
|
|
|
import WebSocketClient from '@gamestdio/websocket';
|
2017-02-04 00:34:31 +01:00
|
|
|
|
2018-05-18 02:32:35 +02:00
|
|
|
const randomIntUpTo = max => Math.floor(Math.random() * Math.floor(max));
|
|
|
|
|
2020-01-21 18:57:21 +01:00
|
|
|
const knownEventTypes = [
|
|
|
|
'update',
|
|
|
|
'delete',
|
|
|
|
'notification',
|
|
|
|
'conversation',
|
|
|
|
'filters_changed',
|
|
|
|
];
|
|
|
|
|
2019-03-07 22:17:52 +01:00
|
|
|
export function connectStream(path, pollingRefresh = null, callbacks = () => ({ onConnect() {}, onDisconnect() {}, onReceive() {} })) {
|
2017-11-15 16:04:15 +01:00
|
|
|
return (dispatch, getState) => {
|
|
|
|
const streamingAPIBaseURL = getState().getIn(['meta', 'streaming_api_base_url']);
|
|
|
|
const accessToken = getState().getIn(['meta', 'access_token']);
|
2019-03-07 22:17:52 +01:00
|
|
|
const { onConnect, onDisconnect, onReceive } = callbacks(dispatch, getState);
|
2018-05-18 02:32:35 +02:00
|
|
|
|
2017-11-15 16:04:15 +01:00
|
|
|
let polling = null;
|
|
|
|
|
|
|
|
const setupPolling = () => {
|
2018-05-18 02:32:35 +02:00
|
|
|
pollingRefresh(dispatch, () => {
|
|
|
|
polling = setTimeout(() => setupPolling(), 20000 + randomIntUpTo(20000));
|
|
|
|
});
|
2017-11-15 16:04:15 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
const clearPolling = () => {
|
|
|
|
if (polling) {
|
2018-05-18 02:32:35 +02:00
|
|
|
clearTimeout(polling);
|
2017-11-15 16:04:15 +01:00
|
|
|
polling = null;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
const subscription = getStream(streamingAPIBaseURL, accessToken, path, {
|
|
|
|
connected () {
|
|
|
|
if (pollingRefresh) {
|
|
|
|
clearPolling();
|
|
|
|
}
|
2019-03-07 22:17:52 +01:00
|
|
|
|
|
|
|
onConnect();
|
2017-11-15 16:04:15 +01:00
|
|
|
},
|
|
|
|
|
|
|
|
disconnected () {
|
|
|
|
if (pollingRefresh) {
|
2018-05-18 02:32:35 +02:00
|
|
|
polling = setTimeout(() => setupPolling(), randomIntUpTo(40000));
|
2017-11-15 16:04:15 +01:00
|
|
|
}
|
2018-05-18 02:32:35 +02:00
|
|
|
|
2017-11-15 16:04:15 +01:00
|
|
|
onDisconnect();
|
|
|
|
},
|
|
|
|
|
|
|
|
received (data) {
|
|
|
|
onReceive(data);
|
|
|
|
},
|
|
|
|
|
|
|
|
reconnected () {
|
|
|
|
if (pollingRefresh) {
|
|
|
|
clearPolling();
|
|
|
|
pollingRefresh(dispatch);
|
|
|
|
}
|
2019-03-07 22:17:52 +01:00
|
|
|
|
|
|
|
onConnect();
|
2017-11-15 16:04:15 +01:00
|
|
|
},
|
|
|
|
|
|
|
|
});
|
|
|
|
|
|
|
|
const disconnect = () => {
|
|
|
|
if (subscription) {
|
|
|
|
subscription.close();
|
|
|
|
}
|
2018-05-18 02:32:35 +02:00
|
|
|
|
2017-11-15 16:04:15 +01:00
|
|
|
clearPolling();
|
|
|
|
};
|
|
|
|
|
|
|
|
return disconnect;
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-04-15 02:32:42 +02:00
|
|
|
export default function getStream(streamingAPIBaseURL, accessToken, stream, { connected, received, disconnected, reconnected }) {
|
2020-01-21 18:57:21 +01:00
|
|
|
const params = stream.split('&');
|
|
|
|
stream = params.shift();
|
|
|
|
|
|
|
|
if (streamingAPIBaseURL.startsWith('ws')) {
|
|
|
|
params.unshift(`stream=${stream}`);
|
|
|
|
const ws = new WebSocketClient(`${streamingAPIBaseURL}/api/v1/streaming/?${params.join('&')}`, accessToken);
|
|
|
|
|
|
|
|
ws.onopen = connected;
|
|
|
|
ws.onmessage = e => received(JSON.parse(e.data));
|
|
|
|
ws.onclose = disconnected;
|
|
|
|
ws.onreconnect = reconnected;
|
|
|
|
|
|
|
|
return ws;
|
|
|
|
}
|
|
|
|
|
|
|
|
params.push(`access_token=${accessToken}`);
|
|
|
|
const es = new EventSource(`${streamingAPIBaseURL}/api/v1/streaming/${stream}?${params.join('&')}`);
|
|
|
|
|
|
|
|
let firstConnect = true;
|
|
|
|
es.onopen = () => {
|
|
|
|
if (firstConnect) {
|
|
|
|
firstConnect = false;
|
|
|
|
connected();
|
|
|
|
} else {
|
|
|
|
reconnected();
|
|
|
|
}
|
|
|
|
};
|
|
|
|
for (let type of knownEventTypes) {
|
|
|
|
es.addEventListener(type, (e) => {
|
|
|
|
received({
|
|
|
|
event: e.type,
|
|
|
|
payload: e.data,
|
|
|
|
});
|
|
|
|
});
|
|
|
|
}
|
|
|
|
es.onerror = disconnected;
|
2017-02-04 00:34:31 +01:00
|
|
|
|
2020-01-21 18:57:21 +01:00
|
|
|
return es;
|
2017-02-04 00:34:31 +01:00
|
|
|
};
|