use-stomp
Version:
react provider, class decorator, and a hook for websockets using the stomp protocol
181 lines (157 loc) • 20.9 kB
JavaScript
;
var _interopRequireWildcard = require("@babel/runtime/helpers/interopRequireWildcard");
var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault");
require("core-js/modules/es.object.get-own-property-descriptor");
Object.defineProperty(exports, "__esModule", {
value: true
});
exports.default = void 0;
var _typeof2 = _interopRequireDefault(require("@babel/runtime/helpers/typeof"));
var _defineProperty2 = _interopRequireDefault(require("@babel/runtime/helpers/defineProperty"));
var _slicedToArray2 = _interopRequireDefault(require("@babel/runtime/helpers/slicedToArray"));
var _react = _interopRequireWildcard(require("react"));
var _sockjsClient = _interopRequireDefault(require("sockjs-client"));
var _context = _interopRequireDefault(require("./context"));
var _stomp = _interopRequireDefault(require("./stomp"));
function ownKeys(object, enumerableOnly) { var keys = Object.keys(object); if (Object.getOwnPropertySymbols) { var symbols = Object.getOwnPropertySymbols(object); if (enumerableOnly) symbols = symbols.filter(function (sym) { return Object.getOwnPropertyDescriptor(object, sym).enumerable; }); keys.push.apply(keys, symbols); } return keys; }
function _objectSpread(target) { for (var i = 1; i < arguments.length; i++) { var source = arguments[i] != null ? arguments[i] : {}; if (i % 2) { ownKeys(Object(source), true).forEach(function (key) { (0, _defineProperty2.default)(target, key, source[key]); }); } else if (Object.getOwnPropertyDescriptors) { Object.defineProperties(target, Object.getOwnPropertyDescriptors(source)); } else { ownKeys(Object(source)).forEach(function (key) { Object.defineProperty(target, key, Object.getOwnPropertyDescriptor(source, key)); }); } } return target; }
var _default = /*#__PURE__*/_react.default.memo(function (props) {
var _client$current, _client$current2;
var client = (0, _react.useRef)(null);
var _useState = (0, _react.useState)(false),
_useState2 = (0, _slicedToArray2.default)(_useState, 2),
connected = _useState2[0],
setConnected = _useState2[1];
var packageMessage = (0, _react.useCallback)(function (name, callable) {
callable.displayName = name;
Object.defineProperty(callable, "name", _objectSpread(_objectSpread({}, Object.getOwnPropertyDescriptor(callable, "name")), {}, {
value: name
}));
return callable;
}("packageMessage", function (channel, msg, optHeaders) {
if (props.packageMessage) {
return props.packageMessage(channel, msg, optHeaders);
}
try {
return (0, _typeof2.default)(msg) === 'object' && msg !== null ? JSON.stringify(msg) : msg;
} catch (e) {
return msg;
}
}), [props.packageMessage]);
var parseMessage = (0, _react.useCallback)(function (name, callable) {
callable.displayName = name;
Object.defineProperty(callable, "name", _objectSpread(_objectSpread({}, Object.getOwnPropertyDescriptor(callable, "name")), {}, {
value: name
}));
return callable;
}("parseMessage", function (channel, msg) {
if (props.parseMessage) {
return props.parseMessage(channel, msg);
}
try {
var parsed = JSON.parse(msg);
return (0, _typeof2.default)(parsed) === 'object' && parsed !== null && parsed.content ? parsed.content : parsed;
} catch (e) {
return msg;
}
}), [props.parseMessage]);
var onConnected = (0, _react.useCallback)(function (name, callable) {
callable.displayName = name;
Object.defineProperty(callable, "name", _objectSpread(_objectSpread({}, Object.getOwnPropertyDescriptor(callable, "name")), {}, {
value: name
}));
return callable;
}("onConnected", function () {
console.log('%c[use-stomp::connected]', 'color: rgb(95,153,63);');
setConnected(function () {
return true;
});
}), []);
var onDisconnected = (0, _react.useCallback)(function (name, callable) {
callable.displayName = name;
Object.defineProperty(callable, "name", _objectSpread(_objectSpread({}, Object.getOwnPropertyDescriptor(callable, "name")), {}, {
value: name
}));
return callable;
}("onDisconnected", function () {
console.log('%c[use-stomp::disconnected]', 'color: rgb(170,34,23);');
setConnected(function () {
return false;
});
client.current = null;
}), []);
var onError = (0, _react.useCallback)(function (name, callable) {
callable.displayName = name;
Object.defineProperty(callable, "name", _objectSpread(_objectSpread({}, Object.getOwnPropertyDescriptor(callable, "name")), {}, {
value: name
}));
return callable;
}("onError", function (error) {
console.error('[use-stomp::error]', error);
}), []);
var send = (0, _react.useCallback)(function (name, callable) {
callable.displayName = name;
Object.defineProperty(callable, "name", _objectSpread(_objectSpread({}, Object.getOwnPropertyDescriptor(callable, "name")), {}, {
value: name
}));
return callable;
}("send", function (channel, msg) {
var optHeaders = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : {};
if (connected) {
client.current.send(channel, optHeaders, packageMessage(channel, msg, optHeaders));
} else {
console.warn('[use-stomp::send]', 'cannot send when websocket is not connected');
}
}), [client.current, connected, packageMessage]);
var subscribe = (0, _react.useCallback)(function (name, callable) {
callable.displayName = name;
Object.defineProperty(callable, "name", _objectSpread(_objectSpread({}, Object.getOwnPropertyDescriptor(callable, "name")), {}, {
value: name
}));
return callable;
}("subscribe", function (channel, callback) {
try {
return client.current.subscribe(channel, function (msg) {
var body = parseMessage(channel, msg.body);
callback(body, msg.headers.destination);
if (body && body.status === 'END') {
client.current.disconnect();
}
}, props.subscribeHeaders).unsubscribe;
} catch (e) {
throw Error(e);
}
}), [(_client$current = client.current) === null || _client$current === void 0 ? void 0 : _client$current.disconnect, (_client$current2 = client.current) === null || _client$current2 === void 0 ? void 0 : _client$current2.subscribe, parseMessage, props.subscribeHeaders]);
(0, _react.useEffect)(function () {
var hasHeaders = props.headers || props.authHeader;
if (hasHeaders && props.url && !connected) {
client.current = _stomp.default.over(new _sockjsClient.default(props.url, null, props.options), !!props.debug);
client.current.connect(props.authHeader ? {
Authorization: props.authHeader
} : props.headers, onConnected, onDisconnected, onError);
}
return function () {
if (connected && client.current) {
client.current.disconnect();
}
};
}, [connected, props.authHeader, props.headers, props.url]);
var ctx = (0, _react.useMemo)(function (name, callable) {
callable.displayName = name;
Object.defineProperty(callable, "name", _objectSpread(_objectSpread({}, Object.getOwnPropertyDescriptor(callable, "name")), {}, {
value: name
}));
return callable;
}("ctx", function () {
return {
send: send,
subscribe: subscribe,
connected: connected
};
}), [connected, send, subscribe]);
return /*#__PURE__*/_react.default.createElement(_context.default.Provider, {
value: ctx
}, props.children);
});
exports.default = _default;
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../src/useStompProvider.tsx"],"names":["React","memo","props","client","connected","setConnected","packageMessage","channel","msg","optHeaders","JSON","stringify","e","parseMessage","parsed","parse","content","onConnected","console","log","onDisconnected","current","onError","error","send","warn","subscribe","callback","body","headers","destination","status","disconnect","subscribeHeaders","unsubscribe","Error","hasHeaders","authHeader","url","Stomp","over","SockJS","options","debug","connect","Authorization","ctx","children"],"mappings":"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","sourcesContent":["import React, {useCallback, useEffect, useMemo, useRef, useState} from 'react';\nimport SockJS, {Options} from 'sockjs-client';\nimport type {Client} from 'stompjs';\n\nimport UseStompCtx from './context';\nimport Stomp from './stomp';\n\nexport type UseStompProviderProps = {\n    /**\n     * HTTP to connect\n     */\n    url: string;\n    /**\n     * Add console logs for debugging\n     */\n    debug?: boolean;\n    /**\n     * SockJS Options (https://github.com/sockjs/sockjs-client#sockjs-client-api)\n     */\n    options?: Options;\n    /**\n     * The request auth header will be passed to the server or agent through the STOMP connection frame\n     */\n    authHeader?: string;\n    /**\n     * The request header will be passed to the server or agent through the STOMP connection frame\n     */\n    headers?: Record<string, unknown>;\n    /**\n     * override default parsing of messages\n     */\n    parseMessage?(channel: string, msg: any): any;\n    /**\n     * override default packaging of messages\n     */\n    packageMessage?(channel: string, msg: any, optHeaders: any): any;\n    /**\n     * The request header that will be passed when subscribing to the target\n     */\n    subscribeHeaders?: Record<string, unknown>;\n};\n\nexport default React.memo<UseStompProviderProps>((props) => {\n    const client = useRef<Client>(null);\n\n    const [connected, setConnected] = useState(false);\n\n    const packageMessage = useCallback(\n        (channel, msg, optHeaders) => {\n            if (props.packageMessage) {\n                return props.packageMessage(channel, msg, optHeaders);\n            }\n\n            try {\n                return typeof msg === 'object' && msg !== null\n                    ? JSON.stringify(msg)\n                    : msg;\n            } catch (e) {\n                return msg;\n            }\n        },\n        [props.packageMessage]\n    );\n\n    const parseMessage = useCallback(\n        (channel, msg) => {\n            if (props.parseMessage) {\n                return props.parseMessage(channel, msg);\n            }\n\n            try {\n                const parsed = JSON.parse(msg);\n\n                return typeof parsed === 'object' &&\n                    parsed !== null &&\n                    parsed.content\n                    ? parsed.content\n                    : parsed;\n            } catch (e) {\n                return msg;\n            }\n        },\n        [props.parseMessage]\n    );\n\n    const onConnected = useCallback(() => {\n        console.log('%c[use-stomp::connected]', 'color: rgb(95,153,63);');\n        setConnected(() => true);\n    }, []);\n\n    const onDisconnected = useCallback(() => {\n        console.log('%c[use-stomp::disconnected]', 'color: rgb(170,34,23);');\n        setConnected(() => false);\n        client.current = null;\n    }, []);\n\n    const onError = useCallback((error) => {\n        console.error('[use-stomp::error]', error);\n    }, []);\n\n    const send = useCallback(\n        (channel, msg, optHeaders = {}) => {\n            if (connected) {\n                client.current.send(\n                    channel,\n                    optHeaders,\n                    packageMessage(channel, msg, optHeaders)\n                );\n            } else {\n                console.warn(\n                    '[use-stomp::send]',\n                    'cannot send when websocket is not connected'\n                );\n            }\n        },\n        [client.current, connected, packageMessage]\n    );\n\n    const subscribe = useCallback(\n        (channel, callback) => {\n            try {\n                return client.current.subscribe(\n                    channel,\n                    (msg: any) => {\n                        const body = parseMessage(channel, msg.body);\n\n                        callback(body, msg.headers.destination);\n\n                        if (body && body.status === 'END') {\n                            (client.current.disconnect as any)();\n                        }\n                    },\n                    props.subscribeHeaders\n                ).unsubscribe;\n            } catch (e) {\n                throw Error(e);\n            }\n        },\n        [\n            client.current?.disconnect,\n            client.current?.subscribe,\n            parseMessage,\n            props.subscribeHeaders\n        ]\n    );\n\n    useEffect(() => {\n        const hasHeaders = props.headers || props.authHeader;\n\n        if (hasHeaders && props.url && !connected) {\n            client.current = Stomp.over(\n                new SockJS(props.url, null, props.options),\n                !!props.debug\n            );\n\n            (client.current.connect as any)(\n                props.authHeader\n                    ? {Authorization: props.authHeader}\n                    : props.headers,\n                onConnected,\n                onDisconnected,\n                onError\n            );\n        }\n\n        return () => {\n            if (connected && client.current) {\n                (client.current.disconnect as any)();\n            }\n        };\n    }, [connected, props.authHeader, props.headers, props.url]);\n\n    const ctx = useMemo(\n        () => ({\n            send,\n            subscribe,\n            connected\n        }),\n        [connected, send, subscribe]\n    );\n\n    return (\n        <UseStompCtx.Provider value={ctx}>\n            {props.children}\n        </UseStompCtx.Provider>\n    );\n});\n"]}