├── .babelrc ├── .editorconfig ├── .eslintrc ├── .gitignore ├── .npmrc ├── CHANGELOG.md ├── DEVELOPMENT.md ├── LICENSE ├── README.md ├── lib ├── cjs │ ├── index.js │ ├── inject.js │ └── types.js ├── es │ ├── index.js │ ├── inject.js │ └── types.js └── umd │ ├── react-signalr.js │ ├── react-signalr.min.js │ └── react-signalr.min.js.map ├── package-lock.json ├── package.json ├── src ├── index.js ├── inject.jsx └── types.jsx ├── tools └── babel.preset.js └── webpack.config.js /.babelrc: -------------------------------------------------------------------------------- 1 | { 2 | "presets": [ "./tools/babel.preset" ] 3 | } -------------------------------------------------------------------------------- /.editorconfig: -------------------------------------------------------------------------------- 1 | # EditorConfig is awesome: http://EditorConfig.org 2 | 3 | # top-most EditorConfig file 4 | root = true 5 | 6 | [*] 7 | end_of_line = lf 8 | indent_style = space 9 | indent_size = 2 10 | charset = utf-8 11 | -------------------------------------------------------------------------------- /.eslintrc: -------------------------------------------------------------------------------- 1 | { 2 | "extends": "airbnb", 3 | "plugins": [ 4 | "react" 5 | ], 6 | "parser": "babel-eslint", 7 | "parserOptions": { 8 | "ecmaVersion": 6, 9 | "sourceType": "module", 10 | "ecmaFeatures": { 11 | "jsx": true 12 | } 13 | }, 14 | "globals": { 15 | "document": false, 16 | "escape": false, 17 | "navigator": false, 18 | "unescape": false, 19 | "window": false, 20 | "describe": true, 21 | "before": true, 22 | "it": true, 23 | "expect": true, 24 | "sinon": true 25 | }, 26 | "env": { 27 | "es6": true, 28 | "browser": true, 29 | "node": true, 30 | "jquery": false, 31 | "mocha": true 32 | }, 33 | "rules": { 34 | "import/no-extraneous-dependencies": ["error", { 35 | "devDependencies": true, 36 | "optionalDependencies": true, 37 | "peerDependencies": true}], 38 | "react/prefer-stateless-function": "off", 39 | "no-console": "off" 40 | } 41 | } -------------------------------------------------------------------------------- /.gitignore: -------------------------------------------------------------------------------- 1 | # Logs 2 | logs 3 | *.log 4 | 5 | # Runtime data 6 | pids 7 | *.pid 8 | *.seed 9 | 10 | # Directory for instrumented libs generated by jscoverage/JSCover 11 | lib-cov 12 | 13 | # Coverage directory used by tools like istanbul 14 | coverage 15 | 16 | # Grunt intermediate storage (http://gruntjs.com/creating-plugins#storing-task-files) 17 | .grunt 18 | 19 | # node-waf configuration 20 | .lock-wscript 21 | 22 | # Compiled binary addons (http://nodejs.org/api/addons.html) 23 | build/Release 24 | 25 | # Dependency directory 26 | # https://www.npmjs.org/doc/misc/npm-faq.html#should-i-check-my-node_modules-folder-into-git 27 | node_modules 28 | 29 | # IntelliJ configs 30 | .idea 31 | -------------------------------------------------------------------------------- /.npmrc: -------------------------------------------------------------------------------- 1 | save-exact = true -------------------------------------------------------------------------------- /CHANGELOG.md: -------------------------------------------------------------------------------- 1 | # Changelog 2 | 3 | * In general follow (https://docs.npmjs.com/getting-started/semantic-versioning) versioning. 4 | 5 | ## 3.3.0 6 | * Configuration change: negotiation phase will be skipped from now on 7 | * Rearranged src folder and removed .npmignore in order to fix npm related problems 8 | * Fixed ejs/es package paths in package.json 9 | * Reverted to a previous version of @aspnet/signalr 10 | * Fixed a bug in stopHub fn. 11 | 12 | ## 3.2.2 13 | * Rearranged src folder and removed .npmignore in order to fix npm related problems 14 | 15 | ## 3.2.1 16 | * Fixed ejs/es package paths in package.json 17 | 18 | ## 3.2.0 19 | * Package updates, incl. @aspnet/signalr update from version 1.0.0 to 1.1.0. 20 | 21 | ## 3.1.0 22 | * Implemented methods in hub proxy for adding and removing clients to and from named groups 23 | 24 | ## 3.0.0 25 | * injectSignalR can now be used as a decorator (contains breaking changes) 26 | * Superficial code restructuring, linter fixes, etc. 27 | 28 | ## 2.0.0 29 | * Migrated to .NET Core 2.1 30 | 31 | ## 1.0.0 32 | * Initial release 33 | -------------------------------------------------------------------------------- /DEVELOPMENT.md: -------------------------------------------------------------------------------- 1 | ### Development workflow 2 | * Make a new branch 3 | * Run `npm run hot` to start examples in watch hot module replacement mode 4 | * Open `http://localhost:5555` 5 | 6 | ### Development workflow with project using the package 7 | ##### Link local package to your project 8 | * Run `npm link` at Component root to make your local package linkable 9 | * Run `npm link @opuscapita/react-signalr` at project's dir that's using the component to use local package 10 | ##### Build and watch the package 11 | * Run `npm run watch[:cjs, :es, :umd]` to run dev builds in watch mode 12 | ##### Unlink local package 13 | * Run `npm unlink @opuscapita/react-signalr` at project's dir that's using the component 14 | 15 | ### Preparing the PR 16 | * Reset `docs` and `lib` directories to master branch state `git checkout master -- docs/* lib/*` 17 | * Update `CHANGELOG.md` with your changes under the `` header 18 | * Make a pull request 19 | 20 | ### Preparing for new version 21 | * Use `master` branch 22 | * Add new version header to `CHANGELOG.md` and move everyting from `` there (leave next header empty) 23 | * Make sure `npm test` and `npm run lint` runs without errors 24 | 25 | ### Creating a new release 26 | * Run `npm version [major|minor|patch]` [Info](https://docs.npmjs.com/cli/version) 27 | 28 | ### Publish new version to NPM 29 | * Run `npm publish` 30 | -------------------------------------------------------------------------------- /LICENSE: -------------------------------------------------------------------------------- 1 | MIT License 2 | 3 | Copyright (c) 2018 4 | 5 | Permission is hereby granted, free of charge, to any person obtaining a copy 6 | of this software and associated documentation files (the "Software"), to deal 7 | in the Software without restriction, including without limitation the rights 8 | to use, copy, modify, merge, publish, distribute, sublicense, and/or sell 9 | copies of the Software, and to permit persons to whom the Software is 10 | furnished to do so, subject to the following conditions: 11 | 12 | The above copyright notice and this permission notice shall be included in all 13 | copies or substantial portions of the Software. 14 | 15 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR 16 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, 17 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE 18 | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER 19 | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, 20 | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE 21 | SOFTWARE. 22 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # react-signalr 2 | 3 | ### Description 4 | Higher-Order Component that provides a connection to a SignalR hub. This component adds a hub proxy, that 5 | may be used to register and unregister event listeners, and also to invoke a hub controller and send data to it. 6 | 7 | This component is built using the [SignalR JavaScript client](https://github.com/aspnet/SignalR/tree/release/2.1/clients/ts/signalr) of 8 | the [ASP.NET Core 2.1 Signalr](https://github.com/aspnet/SignalR/tree/release/2.1) product. 9 | 10 | ### Installation 11 | ``` 12 | npm install @opuscapita/react-signalr --save 13 | ``` 14 | 15 | ### Builds 16 | #### UMD 17 | The default build with compiled styles in the .js file. Also minified version available in the lib/umd directory. 18 | #### CommonJS/ES Module 19 | You need to configure your module loader to use `cjs` or `es` fields of the package.json to use these module types. 20 | Also you need to configure sass loader, since all the styles are in sass format. 21 | * With webpack use [resolve.mainFields](https://webpack.js.org/configuration/resolve/#resolve-mainfields) to configure the module type. 22 | * Add [SASS loader](https://github.com/webpack-contrib/sass-loader) to support importing of SASS styles. 23 | 24 | ### API 25 | 26 | #### Options for ```injectSignalR``` 27 | | Option | Type | Default | Description | 28 | | ------------------------ | ---------------- | ------------------------ | -------------------------------------------------------- | 29 | | hubName | string | required | Name of the signalr hub | 30 | | baseAddress | string \| func | required | Base address for signalr server | 31 | | accessToken | string \| func | | Access token for authorization on the server | 32 | | signalrPath | string | 'signalr' | Path to signalr hubs | 33 | | controller | string | <hubName> | Name of the controller (if different from hubName) | 34 | | retries | integer | 3 | Number of retries to connect after a failure | 35 | 36 | #### Methods in hub proxy 37 | | Method | Parameters | Description | 38 | | ---------------------------- | --------------------------------- | ---------------------------------------------- | 39 | | invoke(target, query) | target: string, query: string | Invokes hub controller with GET | 40 | | send(target, payload) | target: string, payload: object | Invokes hub controller with POST | 41 | | register(event, listener) | event: string, listener: func | Registers a listener for an event | 42 | | unregister(event, listener) | event: string, listener: func | Unregisters a listener for an event | 43 | | add(group) | group: string | Adds client to a named group1) | 44 | | remove(group) | group: string | Removes client from a named group1) | 45 | 46 | 1) To be able to use group messaging the SignalR hub must implement two methods, `AddToGroup(string group)` and `RemoveFromGroup(string group)`, which respectively add and remove the client to and from the specified named group *(cf. [code example](#addingremoving-the-client-tofrom-a-named-group))*. 47 | 48 | ### Code example 49 | 50 | #### Injecting signalr HOC to a component 51 | 52 | ##### As a HOC 53 | ```jsx 54 | import React from 'react'; 55 | import { injectSignalR, hubShape } from '@opuscapita/react-signalr'; 56 | 57 | class MyComponent extends React.Component { 58 | // ... 59 | } 60 | 61 | MyComponent.propTypes = { 62 | // PropType for the hub proxy. 63 | mynotifier: hubShape, 64 | }; 65 | 66 | export default injectSignalR({ 67 | // Defines both the last part of the route to the hub, 68 | // and also the key of the hub proxy in this.props. 69 | // In this case it hub proxy is found in this.props.mynotifier. 70 | hubName: 'mynotifier', 71 | // Either 1) a string containing the server url, or 72 | // 2) a function getting the server url from the state (example). 73 | baseAddress: (state) => state.configuration.server, 74 | // 1) A string containing the access token, or 75 | // 2) a function getting the access token from the state (example), or 76 | // 3) a function using the state to return a function that 77 | // gets the access token. 78 | accessToken: (state) => state.configuration.accessToken, 79 | })(MyComponent); 80 | ``` 81 | 82 | ##### As a decorator 83 | ```jsx 84 | import React from 'react'; 85 | import { injectSignalR, hubShape } from '@opuscapita/react-signalr'; 86 | 87 | @injectSignalR({ 88 | hubName: 'mynotifier', 89 | baseAddress: (state) => state.configuration.server, 90 | accessToken: (state) => state.configuration.accessToken, 91 | }) 92 | export default class MyComponent extends React.Component { 93 | // ... 94 | } 95 | 96 | MyComponent.propTypes = { 97 | // PropType for the hub proxy. 98 | mynotifier: hubShape, 99 | }; 100 | ``` 101 | 102 | #### Passing the hub proxy to child component(s) 103 | ```jsx 104 | class MyComponent extends React.Component { 105 | 106 | // ... 107 | 108 | render() { 109 | // Passing the hub proxy from this.props to child component(s) allows 110 | // also the child component(s) to register its (their) own listeners. 111 | const { ...passThroughProps } = this.props; 112 | return (); 114 | } 115 | } 116 | ``` 117 | 118 | #### Registering and unregistering listeners 119 | ```jsx 120 | class MyComponent extends React.Component { 121 | 122 | // ... 123 | 124 | // Listeners may be registered in componentDidMount (recommended). 125 | componentDidMount() { 126 | // Hub proxy is found in this.props. 127 | const { mynotifier } = this.props; 128 | if (mynotifier) { 129 | // Register this.onInserted to listen 'inserted' event. 130 | mynotifier.register('inserted', this.onInserted); 131 | // Register this.onUpdated to listen 'updated' event. 132 | mynotifier.register('updated', this.onUpdated); 133 | } 134 | } 135 | 136 | // Listeners may be unregistered in componentWillUnmount (recommended). 137 | componentWillUnmount() { 138 | const { mynotifier } = this.props; 139 | if (mynotifier) { 140 | // Unregister this.onInserted from listening to 'inserted' event. 141 | mynotifier.unregister('inserted', this.onInserted); 142 | // Unregister this.onUpdated from listening to 'updated' event. 143 | mynotifier.unregister('updated', this.onUpdated); 144 | } 145 | } 146 | 147 | // Parameter list should match the response sent from the server. 148 | onInserted = (target, id) => { 149 | // Handle inserted event ... 150 | } 151 | 152 | onUpdated = (target, id) => { 153 | // Handle updated event ... 154 | } 155 | 156 | // ... 157 | } 158 | ``` 159 | 160 | #### Invoking controller or sending data to it 161 | ```jsx 162 | class MyComponent extends React.Component { 163 | 164 | // ... 165 | 166 | invoke() { 167 | // Requests '//target/123' with GET 168 | this.props.mynotifier.invoke('target', 123); 169 | } 170 | 171 | send(data) { 172 | // Requests '//target' with POST 173 | // and the JS object `data` as payload in JSON format 174 | this.props.mynotifier.send('target', data); 175 | } 176 | 177 | // ... 178 | } 179 | ``` 180 | 181 | #### Adding/removing the client to/from a named group 182 | 183 | ##### The SignalR hub 184 | ```csharp 185 | public class MyNotifier : Hub 186 | { 187 | // ... 188 | 189 | public Task AddToGroup(string group) 190 | => Groups.AddToGroupAsync(group); 191 | 192 | public Task RemoveFromGroup(string group) 193 | => Groups.RemoveFromGroupAsync(group); 194 | 195 | // ... 196 | } 197 | ``` 198 | 199 | ##### The client component 200 | ```jsx 201 | class MyComponent extends React.Component { 202 | 203 | // ... 204 | 205 | add(group) { 206 | this.props.mynotifier.add(group); 207 | } 208 | 209 | remove(group) { 210 | this.props.mynotifier.remove(group); 211 | } 212 | 213 | // ... 214 | } 215 | ``` 216 | -------------------------------------------------------------------------------- /lib/cjs/index.js: -------------------------------------------------------------------------------- 1 | 'use strict'; 2 | 3 | exports.__esModule = true; 4 | 5 | var _types = require('./types'); 6 | 7 | Object.defineProperty(exports, 'hubShape', { 8 | enumerable: true, 9 | get: function get() { 10 | return _interopRequireDefault(_types).default; 11 | } 12 | }); 13 | 14 | var _inject = require('./inject'); 15 | 16 | Object.defineProperty(exports, 'injectSignalR', { 17 | enumerable: true, 18 | get: function get() { 19 | return _interopRequireDefault(_inject).default; 20 | } 21 | }); 22 | 23 | function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } 24 | //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9pbmRleC5qcyJdLCJuYW1lcyI6WyJkZWZhdWx0Il0sIm1hcHBpbmdzIjoiOzs7Ozs7Ozs7MENBQVNBLE87Ozs7Ozs7OzsyQ0FDQUEsTyIsImZpbGUiOiJpbmRleC5qcyIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCB7IGRlZmF1bHQgYXMgaHViU2hhcGUgfSBmcm9tICcuL3R5cGVzJztcbmV4cG9ydCB7IGRlZmF1bHQgYXMgaW5qZWN0U2lnbmFsUiB9IGZyb20gJy4vaW5qZWN0JztcbiJdfQ== -------------------------------------------------------------------------------- /lib/cjs/inject.js: -------------------------------------------------------------------------------- 1 | 'use strict'; 2 | 3 | exports.__esModule = true; 4 | 5 | var _extends = Object.assign || function (target) { for (var i = 1; i < arguments.length; i++) { var source = arguments[i]; for (var key in source) { if (Object.prototype.hasOwnProperty.call(source, key)) { target[key] = source[key]; } } } return target; }; 6 | 7 | var _react = require('react'); 8 | 9 | var _react2 = _interopRequireDefault(_react); 10 | 11 | var _propTypes = require('prop-types'); 12 | 13 | var _propTypes2 = _interopRequireDefault(_propTypes); 14 | 15 | var _axios = require('axios'); 16 | 17 | var _axios2 = _interopRequireDefault(_axios); 18 | 19 | var _redux = require('redux'); 20 | 21 | var _reactRedux = require('react-redux'); 22 | 23 | var _immutable = require('immutable'); 24 | 25 | var _signalr = require('@aspnet/signalr'); 26 | 27 | function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } 28 | 29 | function _objectWithoutProperties(obj, keys) { var target = {}; for (var i in obj) { if (keys.indexOf(i) >= 0) continue; if (!Object.prototype.hasOwnProperty.call(obj, i)) continue; target[i] = obj[i]; } return target; } 30 | 31 | function _asyncToGenerator(fn) { return function () { var gen = fn.apply(this, arguments); return new Promise(function (resolve, reject) { function step(key, arg) { try { var info = gen[key](arg); var value = info.value; } catch (error) { reject(error); return; } if (info.done) { resolve(value); } else { return Promise.resolve(value).then(function (value) { step("next", value); }, function (err) { step("throw", err); }); } } return step("next"); }); }; } 32 | 33 | function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } } 34 | 35 | function _possibleConstructorReturn(self, call) { if (!self) { throw new ReferenceError("this hasn't been initialised - super() hasn't been called"); } return call && (typeof call === "object" || typeof call === "function") ? call : self; } 36 | 37 | function _inherits(subClass, superClass) { if (typeof superClass !== "function" && superClass !== null) { throw new TypeError("Super expression must either be null or a function, not " + typeof superClass); } subClass.prototype = Object.create(superClass && superClass.prototype, { constructor: { value: subClass, enumerable: false, writable: true, configurable: true } }); if (superClass) Object.setPrototypeOf ? Object.setPrototypeOf(subClass, superClass) : subClass.__proto__ = superClass; } 38 | 39 | var getDisplayName = function getDisplayName(Component) { 40 | return Component.displayName || Component.name || 'Component'; 41 | }; 42 | 43 | var injectSignalR = function injectSignalR(options) { 44 | return function (WrappedComponent) { 45 | var _class, _temp; 46 | 47 | var _options$hubName = options.hubName, 48 | hubName = _options$hubName === undefined ? '' : _options$hubName, 49 | _options$baseAddress = options.baseAddress, 50 | baseAddress = _options$baseAddress === undefined ? 'http://localhost:5555' : _options$baseAddress, 51 | _options$accessToken = options.accessToken, 52 | accessToken = _options$accessToken === undefined ? null : _options$accessToken, 53 | _options$signalrPath = options.signalrPath, 54 | signalrPath = _options$signalrPath === undefined ? 'signalr' : _options$signalrPath, 55 | _options$retries = options.retries, 56 | retries = _options$retries === undefined ? 3 : _options$retries; 57 | var _options$controller = options.controller, 58 | controller = _options$controller === undefined ? hubName : _options$controller; 59 | var InjectSignalR = (_temp = _class = function (_React$PureComponent) { 60 | _inherits(InjectSignalR, _React$PureComponent); 61 | 62 | function InjectSignalR(props) { 63 | _classCallCheck(this, InjectSignalR); 64 | 65 | var _this = _possibleConstructorReturn(this, _React$PureComponent.call(this, props)); 66 | 67 | _this.count = function (c, s) { 68 | return c + s.count(); 69 | }; 70 | 71 | _this.addToGroup = function (group) { 72 | var hub = _this.state.hub; 73 | 74 | if (hub) { 75 | var connection = hub.connection; 76 | 77 | if (connection && connection.connectionState === 1) { 78 | hub.invoke('addToGroup', group).catch(function (err) { 79 | console.error('Error: Adding client to group ' + group + ' in ' + hubName + ' failed.\n\n' + err); 80 | }); 81 | } 82 | } 83 | }; 84 | 85 | _this.removeFromGroup = function (group) { 86 | var hub = _this.state.hub; 87 | 88 | if (hub) { 89 | var connection = hub.connection; 90 | 91 | if (connection && connection.connectionState === 1) { 92 | return hub.invoke('removeFromGroup', group).catch(function (err) { 93 | console.error('Error: Removing client from group ' + group + ' in ' + hubName + ' failed.\n\n' + err); 94 | }); 95 | } 96 | } 97 | return Promise.resolve(); 98 | }; 99 | 100 | _this.sendToController = function (target) { 101 | var data = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : null; 102 | 103 | var url = _this.props.baseUrl + '/' + controller + '/' + target; 104 | var payload = data ? data.toJS() : null; 105 | return _axios2.default.post(url, payload).catch(function (err) { 106 | console.error('Error: Sending data to ' + controller + ' failed.\n\n' + err); 107 | }); 108 | }; 109 | 110 | _this.invokeController = function (targetMethod) { 111 | var data = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : null; 112 | 113 | var urlBase = _this.props.baseUrl + '/' + controller + '/' + targetMethod; 114 | var url = data ? urlBase + '/' + data : urlBase; 115 | return _axios2.default.get(url).catch(function (err) { 116 | console.error('Error: Invoking ' + controller + ' failed.\n\n' + err); 117 | }); 118 | }; 119 | 120 | _this.handleError = function (err) { 121 | var response = err.response, 122 | statusCode = err.statusCode; 123 | 124 | var _ref = response || {}, 125 | status = _ref.status; 126 | 127 | switch (status || statusCode) { 128 | case 500: 129 | break; 130 | case 401: 131 | _this.oldToken = _this.token; // fall through 132 | default: 133 | _this.setState({ hub: null }); 134 | break; 135 | } 136 | }; 137 | 138 | _this.registerListener = function (name, handler) { 139 | var _this$state = _this.state, 140 | pending = _this$state.pending, 141 | active = _this$state.active, 142 | moribund = _this$state.moribund; 143 | // Remove listener from moribund listeners 144 | 145 | if (!_this.moribund) _this.moribund = moribund || (0, _immutable.Map)(); 146 | var existingMoribund = _this.moribund.getIn([name], (0, _immutable.Set)()); 147 | if (existingMoribund.has(handler)) { 148 | var remainingMoribund = existingMoribund.filterNot(function (h) { 149 | return h === handler; 150 | }); 151 | _this.moribund = remainingMoribund.size ? _this.moribund.setIn([name], remainingMoribund) : _this.moribund.delete(name); 152 | } 153 | // Add listener to pending listeners (if it is NOT active) 154 | if (!_this.active) _this.active = active || (0, _immutable.Map)(); 155 | var existingActive = _this.active.getIn([name], (0, _immutable.Set)()); 156 | if (!existingActive.has(handler)) { 157 | if (!_this.pending) _this.pending = pending || (0, _immutable.Map)(); 158 | var existingPending = _this.pending.getIn([name], (0, _immutable.Set)()); 159 | if (!existingPending.has(handler)) { 160 | _this.pending = _this.pending.setIn([name], existingPending.add(handler)); 161 | } 162 | } 163 | if (_this.pending !== pending || _this.moribund !== moribund) { 164 | _this.setState({ 165 | pending: _this.pending, 166 | moribund: _this.moribund 167 | }); 168 | } 169 | }; 170 | 171 | _this.unregisterListener = function (name, handler) { 172 | var _this$state2 = _this.state, 173 | pending = _this$state2.pending, 174 | active = _this$state2.active, 175 | moribund = _this$state2.moribund; 176 | // Remove listener from pending listeners 177 | 178 | if (!_this.pending) _this.pending = pending || (0, _immutable.Map)(); 179 | var existingPending = _this.pending.getIn([name], (0, _immutable.Set)()); 180 | if (existingPending.has(handler)) { 181 | var remainingPending = existingPending.filterNot(function (h) { 182 | return h === handler; 183 | }); 184 | _this.pending = remainingPending.count() ? _this.pending.setIn([name], remainingPending) : _this.pending.delete(name); 185 | } 186 | // Add listener to moribund listeners (if it is active) 187 | if (!_this.active) _this.active = active || (0, _immutable.Map)(); 188 | var existingActive = _this.active.getIn([name], (0, _immutable.Set)()); 189 | if (existingActive.has(handler)) { 190 | if (!_this.moribund) _this.moribund = moribund || (0, _immutable.Map)(); 191 | var existingMoribund = _this.moribund.getIn([name], (0, _immutable.Set)()); 192 | if (!existingMoribund.has(handler)) { 193 | _this.moribund = _this.moribund.setIn([name], existingMoribund.add(handler)); 194 | } 195 | } 196 | if (_this.pending !== pending || _this.moribund !== moribund) { 197 | _this.setState({ 198 | pending: _this.pending, 199 | moribund: _this.moribund 200 | }); 201 | } 202 | }; 203 | 204 | _this.state = { 205 | hub: null, 206 | pending: undefined, 207 | active: undefined, 208 | moribund: undefined, 209 | retry: 0, 210 | create: 0 211 | }; 212 | return _this; 213 | } 214 | 215 | InjectSignalR.prototype.componentWillMount = function componentWillMount() { 216 | this.hubProxy = { 217 | send: this.sendToController, 218 | invoke: this.invokeController, 219 | add: this.addToGroup, 220 | remove: this.removeFromGroup, 221 | connectionId: undefined, 222 | register: this.registerListener, 223 | unregister: this.unregisterListener 224 | }; 225 | }; 226 | 227 | InjectSignalR.prototype.componentDidMount = function componentDidMount() { 228 | this.createHub(); 229 | }; 230 | 231 | InjectSignalR.prototype.componentWillUpdate = function componentWillUpdate(nextProps, nextState) { 232 | if (this.state.hub !== nextState.hub) { 233 | if (this.state.hub) this.stopHub(this.state.hub, false); 234 | if (nextState.hub) { 235 | this.startHub(nextState.hub); 236 | } else { 237 | this.createHub(nextState.create); 238 | } 239 | } else if (!nextState.hub) { 240 | this.createHub(nextState.create); 241 | } else { 242 | var pending = nextState.pending, 243 | moribund = nextState.moribund; 244 | 245 | if (!moribund) { 246 | moribund = this.moribund || (0, _immutable.Map)(); 247 | } else if (this.moribund) { 248 | moribund = moribund.mergeDeep(this.moribund); 249 | } 250 | var moribundCount = moribund.reduce(this.count, 0); 251 | if (moribundCount) { 252 | this.moribund = this.inactivateListeners(this.state.hub, moribund); 253 | } 254 | if (!pending) { 255 | pending = this.pending || (0, _immutable.Map)(); 256 | } else if (this.pending) { 257 | pending = pending.mergeDeep(this.pending); 258 | } 259 | var pendingCount = pending.reduce(this.count, 0); 260 | if (pendingCount) { 261 | this.pending = this.activateListeners(nextState.hub, pending); 262 | } 263 | } 264 | }; 265 | 266 | InjectSignalR.prototype.componentWillUnmount = function componentWillUnmount() { 267 | this.stopHub(this.state.hub, true); 268 | }; 269 | 270 | InjectSignalR.prototype.createHub = function () { 271 | var _ref2 = _asyncToGenerator( /*#__PURE__*/regeneratorRuntime.mark(function _callee(curCreate) { 272 | var _this2 = this; 273 | 274 | var _state, retry, create, _props, baseUrl, signalrActions, hubAddress, hub; 275 | 276 | return regeneratorRuntime.wrap(function _callee$(_context) { 277 | while (1) { 278 | switch (_context.prev = _context.next) { 279 | case 0: 280 | _state = this.state, retry = _state.retry, create = _state.create; 281 | 282 | if (!(retry > retries)) { 283 | _context.next = 6; 284 | break; 285 | } 286 | 287 | console.error('Error: Ran out of retries for starting ' + hubName + '!'); 288 | this.setState({ 289 | retry: 0, 290 | create: 0 291 | }); 292 | _context.next = 20; 293 | break; 294 | 295 | case 6: 296 | _props = this.props, baseUrl = _props.baseUrl, signalrActions = _props.signalrActions; 297 | 298 | if (!(baseUrl && hubName)) { 299 | _context.next = 20; 300 | break; 301 | } 302 | 303 | hubAddress = baseUrl; 304 | 305 | if (signalrPath) hubAddress = hubAddress + '/' + signalrPath; 306 | hubAddress = hubAddress + '/' + hubName; 307 | this.token = signalrActions.accessTokenFactory(accessToken); 308 | 309 | if (!this.token) { 310 | _context.next = 17; 311 | break; 312 | } 313 | 314 | if (!(this.oldToken === this.token)) { 315 | _context.next = 16; 316 | break; 317 | } 318 | 319 | if ((curCreate || create) > retries) { 320 | console.warn('Warning: Unable to get up-to-date access token.'); 321 | } else { 322 | this.setState({ 323 | hub: null, 324 | create: (curCreate || create) + 1 325 | }); 326 | } 327 | return _context.abrupt('return'); 328 | 329 | case 16: 330 | this.oldToken = undefined; 331 | 332 | case 17: 333 | hub = new _signalr.HubConnectionBuilder().withUrl(hubAddress, { 334 | skipNegotiation: true, 335 | transport: _signalr.HttpTransportType.WebSockets, 336 | accessTokenFactory: function accessTokenFactory() { 337 | return _this2.token; 338 | } 339 | }).build(); 340 | 341 | hub.onclose = this.handleError; 342 | this.setState({ 343 | hub: hub, 344 | retry: retry + 1, 345 | create: 0 346 | }); 347 | 348 | case 20: 349 | case 'end': 350 | return _context.stop(); 351 | } 352 | } 353 | }, _callee, this); 354 | })); 355 | 356 | function createHub(_x3) { 357 | return _ref2.apply(this, arguments); 358 | } 359 | 360 | return createHub; 361 | }(); 362 | 363 | InjectSignalR.prototype.startHub = function startHub(hub) { 364 | var _this3 = this; 365 | 366 | if (hub) { 367 | hub.start().then(function () { 368 | var _state2 = _this3.state, 369 | pending = _state2.pending, 370 | active = _state2.active; 371 | 372 | if (!_this3.pending) _this3.pending = pending || (0, _immutable.Map)(); 373 | if (!_this3.active) _this3.active = active || (0, _immutable.Map)(); 374 | _this3.setState({ 375 | active: _this3.active, 376 | pending: _this3.pending, 377 | retry: 0 378 | }); 379 | }).catch(function (err) { 380 | console.warn('Warning: Error while establishing connection to hub ' + hubName + '.\n\n' + err); 381 | hub.stop(); 382 | _this3.handleError(err); 383 | }); 384 | } 385 | }; 386 | 387 | InjectSignalR.prototype.stopHub = function stopHub(hub, clear) { 388 | if (hub) { 389 | var promises = []; 390 | 391 | if (clear) { 392 | // Clear pending 393 | this.pending = undefined; 394 | promises.push(this.removeFromGroup('')); 395 | // Merge active to pending 396 | } else if (!this.pending) { 397 | this.pending = this.state.active; 398 | } else if (this.state.active) { 399 | this.pending = this.pending.mergeDeep(this.state.active); 400 | } 401 | 402 | Promise.all(promises).then(function () { 403 | hub.stop(); 404 | }); 405 | 406 | this.active = undefined; 407 | this.setState({ 408 | pending: this.pending, 409 | active: this.active 410 | }); 411 | } 412 | }; 413 | 414 | InjectSignalR.prototype.activateListeners = function activateListeners(hub, pendingParam) { 415 | var _this4 = this; 416 | 417 | var pending = pendingParam; 418 | if (hub && pendingParam) { 419 | var connection = hub.connection; 420 | 421 | if (connection && connection.connectionState === 1) { 422 | var active = this.state.active; 423 | 424 | if (!this.active) this.active = active || (0, _immutable.Map)(); 425 | if (this.active.reduce(this.count, 0)) { 426 | pending = pending.mapEntries(function (_ref3) { 427 | var name = _ref3[0], 428 | curHandlers = _ref3[1]; 429 | 430 | var existing = _this4.active.getIn([name]); 431 | var handlers = existing ? curHandlers.filterNot(function (handler) { 432 | return existing.has(handler); 433 | }) : curHandlers; 434 | return [name, handlers]; 435 | }); 436 | } 437 | pending.mapEntries(function (_ref4) { 438 | var name = _ref4[0], 439 | handlers = _ref4[1]; 440 | return handlers.map(function (handler) { 441 | return hub.on(name, handler); 442 | }); 443 | }); 444 | this.active = this.active.mergeDeep(pending); 445 | this.setState({ 446 | pending: undefined, 447 | active: this.active 448 | }); 449 | return undefined; 450 | } 451 | } 452 | return pending; 453 | }; 454 | 455 | InjectSignalR.prototype.inactivateListeners = function inactivateListeners(hub, moribund) { 456 | if (hub && moribund) { 457 | moribund.mapEntries(function (_ref5) { 458 | var name = _ref5[0], 459 | handlers = _ref5[1]; 460 | return handlers.map(function (handler) { 461 | return hub.off(name, handler); 462 | }); 463 | }); 464 | var active = this.state.active; 465 | 466 | if (!this.active) this.active = active || (0, _immutable.Map)(); 467 | this.active = this.active.mapEntries(function (_ref6) { 468 | var name = _ref6[0], 469 | curHandlers = _ref6[1]; 470 | 471 | var removable = moribund.getIn([name]); 472 | var handlers = removable ? curHandlers.filterNot(function (handler) { 473 | return removable.has(handler); 474 | }) : curHandlers; 475 | return [name, handlers]; 476 | }); 477 | this.setState({ 478 | active: this.active, 479 | moribund: undefined 480 | }); 481 | return undefined; 482 | } 483 | return moribund; 484 | }; 485 | 486 | InjectSignalR.prototype.render = function render() { 487 | var _hubProp; 488 | 489 | var _props2 = this.props, 490 | baseUrl = _props2.baseUrl, 491 | signalrActions = _props2.signalrActions, 492 | passThroughProps = _objectWithoutProperties(_props2, ['baseUrl', 'signalrActions']); 493 | 494 | var hubProp = (_hubProp = {}, _hubProp[hubName] = this.hubProxy, _hubProp); 495 | return _react2.default.createElement(WrappedComponent, _extends({}, passThroughProps, hubProp)); 496 | }; 497 | 498 | return InjectSignalR; 499 | }(_react2.default.PureComponent), _class.WrappedComponent = WrappedComponent, _temp); 500 | 501 | 502 | InjectSignalR.displayName = 'InjectSignalR(' + getDisplayName(WrappedComponent) + ')'; 503 | 504 | var getValueFromState = function getValueFromState(state, source) { 505 | if (typeof source === 'function') return source(state); 506 | if (typeof source === 'string') return source; 507 | return ''; 508 | }; 509 | 510 | var mapDispatchToProps = function mapDispatchToProps(dispatch) { 511 | return { 512 | signalrActions: (0, _redux.bindActionCreators)({ 513 | accessTokenFactory: function accessTokenFactory() { 514 | return function (dispatcher, getState) { 515 | var state = getState(); 516 | return getValueFromState(state, accessToken); 517 | }; 518 | } 519 | }, dispatch) 520 | }; 521 | }; 522 | 523 | var mapStateToProps = function mapStateToProps(state) { 524 | var baseUrl = getValueFromState(state, baseAddress); 525 | return { baseUrl: baseUrl }; 526 | }; 527 | 528 | return (0, _reactRedux.connect)(mapStateToProps, mapDispatchToProps)(InjectSignalR); 529 | }; 530 | }; 531 | 532 | exports.default = injectSignalR; 533 | //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/inject.jsx"],"names":["getDisplayName","Component","displayName","name","injectSignalR","WrappedComponent","options","hubName","baseAddress","accessToken","signalrPath","retries","controller","InjectSignalR","props","count","c","s","addToGroup","group","hub","state","connection","connectionState","invoke","catch","err","console","error","removeFromGroup","Promise","resolve","sendToController","target","data","url","baseUrl","payload","toJS","axios","post","invokeController","targetMethod","urlBase","get","handleError","response","statusCode","status","oldToken","token","setState","registerListener","handler","pending","active","moribund","existingMoribund","getIn","has","remainingMoribund","filterNot","h","size","setIn","delete","existingActive","existingPending","add","unregisterListener","remainingPending","undefined","retry","create","componentWillMount","hubProxy","send","remove","connectionId","register","unregister","componentDidMount","createHub","componentWillUpdate","nextProps","nextState","stopHub","startHub","mergeDeep","moribundCount","reduce","inactivateListeners","pendingCount","activateListeners","componentWillUnmount","curCreate","signalrActions","hubAddress","accessTokenFactory","warn","HubConnectionBuilder","withUrl","skipNegotiation","transport","HttpTransportType","WebSockets","build","onclose","start","then","stop","clear","promises","push","all","pendingParam","mapEntries","curHandlers","existing","handlers","map","on","off","removable","render","passThroughProps","hubProp","React","PureComponent","getValueFromState","source","mapDispatchToProps","dispatcher","getState","dispatch","mapStateToProps"],"mappings":";;;;;;AAAA;;;;AACA;;;;AACA;;;;AACA;;AACA;;AACA;;AACA;;;;;;;;;;;;;;AAEA,IAAMA,iBAAiB,SAAjBA,cAAiB;AAAA,SAAaC,UAAUC,WAAV,IAAyBD,UAAUE,IAAnC,IAA2C,WAAxD;AAAA,CAAvB;;AAEA,IAAMC,gBAAgB,SAAhBA,aAAgB;AAAA,SAAW,UAACC,gBAAD,EAAsB;AAAA;;AAAA,2BAOjDC,OAPiD,CAEnDC,OAFmD;AAAA,QAEnDA,OAFmD,oCAEzC,EAFyC;AAAA,+BAOjDD,OAPiD,CAGnDE,WAHmD;AAAA,QAGnDA,WAHmD,wCAGrC,uBAHqC;AAAA,+BAOjDF,OAPiD,CAInDG,WAJmD;AAAA,QAInDA,WAJmD,wCAIrC,IAJqC;AAAA,+BAOjDH,OAPiD,CAKnDI,WALmD;AAAA,QAKnDA,WALmD,wCAKrC,SALqC;AAAA,2BAOjDJ,OAPiD,CAMnDK,OANmD;AAAA,QAMnDA,OANmD,oCAMzC,CANyC;AAAA,8BAQpBL,OARoB,CAQ7CM,UAR6C;AAAA,QAQ7CA,UAR6C,uCAQhCL,OARgC;AAAA,QAU/CM,aAV+C;AAAA;;AAanD,6BAAYC,KAAZ,EAAmB;AAAA;;AAAA,qDACjB,gCAAMA,KAAN,CADiB;;AAAA,cAiEnBC,KAjEmB,GAiEX,UAACC,CAAD,EAAIC,CAAJ;AAAA,iBAAUD,IAAIC,EAAEF,KAAF,EAAd;AAAA,SAjEW;;AAAA,cAmEnBG,UAnEmB,GAmEN,UAACC,KAAD,EAAW;AAAA,cACdC,GADc,GACN,MAAKC,KADC,CACdD,GADc;;AAEtB,cAAIA,GAAJ,EAAS;AAAA,gBACCE,UADD,GACgBF,GADhB,CACCE,UADD;;AAEP,gBAAIA,cAAcA,WAAWC,eAAX,KAA+B,CAAjD,EAAoD;AAClDH,kBAAII,MAAJ,CAAW,YAAX,EAAyBL,KAAzB,EACGM,KADH,CACS,UAACC,GAAD,EAAS;AACdC,wBAAQC,KAAR,oCAA+CT,KAA/C,YAA2DZ,OAA3D,oBAAiFmB,GAAjF;AACD,eAHH;AAID;AACF;AACF,SA9EkB;;AAAA,cAgFnBG,eAhFmB,GAgFD,UAACV,KAAD,EAAW;AAAA,cACnBC,GADmB,GACX,MAAKC,KADM,CACnBD,GADmB;;AAE3B,cAAIA,GAAJ,EAAS;AAAA,gBACCE,UADD,GACgBF,GADhB,CACCE,UADD;;AAEP,gBAAIA,cAAcA,WAAWC,eAAX,KAA+B,CAAjD,EAAoD;AAClD,qBAAOH,IAAII,MAAJ,CAAW,iBAAX,EAA8BL,KAA9B,EACJM,KADI,CACE,UAACC,GAAD,EAAS;AACdC,wBAAQC,KAAR,wCAAmDT,KAAnD,YAA+DZ,OAA/D,oBAAqFmB,GAArF;AACD,eAHI,CAAP;AAID;AACF;AACD,iBAAOI,QAAQC,OAAR,EAAP;AACD,SA5FkB;;AAAA,cA8FnBC,gBA9FmB,GA8FA,UAACC,MAAD,EAAyB;AAAA,cAAhBC,IAAgB,uEAAT,IAAS;;AAC1C,cAAMC,MAAS,MAAKrB,KAAL,CAAWsB,OAApB,SAA+BxB,UAA/B,SAA6CqB,MAAnD;AACA,cAAMI,UAAUH,OAAOA,KAAKI,IAAL,EAAP,GAAqB,IAArC;AACA,iBAAOC,gBAAMC,IAAN,CAAWL,GAAX,EAAgBE,OAAhB,EACJZ,KADI,CACE,UAACC,GAAD,EAAS;AACdC,oBAAQC,KAAR,6BAAwChB,UAAxC,oBAAiEc,GAAjE;AACD,WAHI,CAAP;AAID,SArGkB;;AAAA,cAuGnBe,gBAvGmB,GAuGA,UAACC,YAAD,EAA+B;AAAA,cAAhBR,IAAgB,uEAAT,IAAS;;AAChD,cAAMS,UAAa,MAAK7B,KAAL,CAAWsB,OAAxB,SAAmCxB,UAAnC,SAAiD8B,YAAvD;AACA,cAAMP,MAAMD,OAAUS,OAAV,SAAqBT,IAArB,GAA8BS,OAA1C;AACA,iBAAOJ,gBAAMK,GAAN,CAAUT,GAAV,EACJV,KADI,CACE,UAACC,GAAD,EAAS;AACdC,oBAAQC,KAAR,sBAAiChB,UAAjC,oBAA0Dc,GAA1D;AACD,WAHI,CAAP;AAID,SA9GkB;;AAAA,cAmLnBmB,WAnLmB,GAmLL,UAACnB,GAAD,EAAS;AAAA,cACboB,QADa,GACYpB,GADZ,CACboB,QADa;AAAA,cACHC,UADG,GACYrB,GADZ,CACHqB,UADG;;AAAA,qBAEFD,YAAY,EAFV;AAAA,cAEbE,MAFa,QAEbA,MAFa;;AAGrB,kBAAQA,UAAUD,UAAlB;AACE,iBAAK,GAAL;AACE;AACF,iBAAK,GAAL;AACE,oBAAKE,QAAL,GAAgB,MAAKC,KAArB,CAJJ,CAIgC;AAC9B;AACE,oBAAKC,QAAL,CAAc,EAAE/B,KAAK,IAAP,EAAd;AACA;AAPJ;AASD,SA/LkB;;AAAA,cA4NnBgC,gBA5NmB,GA4NA,UAACjD,IAAD,EAAOkD,OAAP,EAAmB;AAAA,4BACE,MAAKhC,KADP;AAAA,cAC5BiC,OAD4B,eAC5BA,OAD4B;AAAA,cACnBC,MADmB,eACnBA,MADmB;AAAA,cACXC,QADW,eACXA,QADW;AAEpC;;AACA,cAAI,CAAC,MAAKA,QAAV,EAAoB,MAAKA,QAAL,GAAgBA,YAAY,qBAA5B;AACpB,cAAMC,mBAAmB,MAAKD,QAAL,CAAcE,KAAd,CAAoB,CAACvD,IAAD,CAApB,EAA4B,qBAA5B,CAAzB;AACA,cAAIsD,iBAAiBE,GAAjB,CAAqBN,OAArB,CAAJ,EAAmC;AACjC,gBAAMO,oBAAoBH,iBAAiBI,SAAjB,CAA2B;AAAA,qBAAKC,MAAMT,OAAX;AAAA,aAA3B,CAA1B;AACA,kBAAKG,QAAL,GAAgBI,kBAAkBG,IAAlB,GACZ,MAAKP,QAAL,CAAcQ,KAAd,CAAoB,CAAC7D,IAAD,CAApB,EAA4ByD,iBAA5B,CADY,GACqC,MAAKJ,QAAL,CAAcS,MAAd,CAAqB9D,IAArB,CADrD;AAED;AACD;AACA,cAAI,CAAC,MAAKoD,MAAV,EAAkB,MAAKA,MAAL,GAAcA,UAAU,qBAAxB;AAClB,cAAMW,iBAAiB,MAAKX,MAAL,CAAYG,KAAZ,CAAkB,CAACvD,IAAD,CAAlB,EAA0B,qBAA1B,CAAvB;AACA,cAAI,CAAC+D,eAAeP,GAAf,CAAmBN,OAAnB,CAAL,EAAkC;AAChC,gBAAI,CAAC,MAAKC,OAAV,EAAmB,MAAKA,OAAL,GAAeA,WAAW,qBAA1B;AACnB,gBAAMa,kBAAkB,MAAKb,OAAL,CAAaI,KAAb,CAAmB,CAACvD,IAAD,CAAnB,EAA2B,qBAA3B,CAAxB;AACA,gBAAI,CAACgE,gBAAgBR,GAAhB,CAAoBN,OAApB,CAAL,EAAmC;AACjC,oBAAKC,OAAL,GAAe,MAAKA,OAAL,CAAaU,KAAb,CAAmB,CAAC7D,IAAD,CAAnB,EAA2BgE,gBAAgBC,GAAhB,CAAoBf,OAApB,CAA3B,CAAf;AACD;AACF;AACD,cAAI,MAAKC,OAAL,KAAiBA,OAAjB,IAA4B,MAAKE,QAAL,KAAkBA,QAAlD,EAA4D;AAC1D,kBAAKL,QAAL,CAAc;AACZG,uBAAS,MAAKA,OADF;AAEZE,wBAAU,MAAKA;AAFH,aAAd;AAID;AACF,SAtPkB;;AAAA,cAwPnBa,kBAxPmB,GAwPE,UAAClE,IAAD,EAAOkD,OAAP,EAAmB;AAAA,6BACA,MAAKhC,KADL;AAAA,cAC9BiC,OAD8B,gBAC9BA,OAD8B;AAAA,cACrBC,MADqB,gBACrBA,MADqB;AAAA,cACbC,QADa,gBACbA,QADa;AAEtC;;AACA,cAAI,CAAC,MAAKF,OAAV,EAAmB,MAAKA,OAAL,GAAeA,WAAW,qBAA1B;AACnB,cAAMa,kBAAkB,MAAKb,OAAL,CAAaI,KAAb,CAAmB,CAACvD,IAAD,CAAnB,EAA2B,qBAA3B,CAAxB;AACA,cAAIgE,gBAAgBR,GAAhB,CAAoBN,OAApB,CAAJ,EAAkC;AAChC,gBAAMiB,mBAAmBH,gBAAgBN,SAAhB,CAA0B;AAAA,qBAAKC,MAAMT,OAAX;AAAA,aAA1B,CAAzB;AACA,kBAAKC,OAAL,GAAegB,iBAAiBvD,KAAjB,KACX,MAAKuC,OAAL,CAAaU,KAAb,CAAmB,CAAC7D,IAAD,CAAnB,EAA2BmE,gBAA3B,CADW,GAEX,MAAKhB,OAAL,CAAaW,MAAb,CAAoB9D,IAApB,CAFJ;AAGD;AACD;AACA,cAAI,CAAC,MAAKoD,MAAV,EAAkB,MAAKA,MAAL,GAAcA,UAAU,qBAAxB;AAClB,cAAMW,iBAAiB,MAAKX,MAAL,CAAYG,KAAZ,CAAkB,CAACvD,IAAD,CAAlB,EAA0B,qBAA1B,CAAvB;AACA,cAAI+D,eAAeP,GAAf,CAAmBN,OAAnB,CAAJ,EAAiC;AAC/B,gBAAI,CAAC,MAAKG,QAAV,EAAoB,MAAKA,QAAL,GAAgBA,YAAY,qBAA5B;AACpB,gBAAMC,mBAAmB,MAAKD,QAAL,CAAcE,KAAd,CAAoB,CAACvD,IAAD,CAApB,EAA4B,qBAA5B,CAAzB;AACA,gBAAI,CAACsD,iBAAiBE,GAAjB,CAAqBN,OAArB,CAAL,EAAoC;AAClC,oBAAKG,QAAL,GAAgB,MAAKA,QAAL,CAAcQ,KAAd,CAAoB,CAAC7D,IAAD,CAApB,EAA4BsD,iBAAiBW,GAAjB,CAAqBf,OAArB,CAA5B,CAAhB;AACD;AACF;AACD,cAAI,MAAKC,OAAL,KAAiBA,OAAjB,IAA4B,MAAKE,QAAL,KAAkBA,QAAlD,EAA4D;AAC1D,kBAAKL,QAAL,CAAc;AACZG,uBAAS,MAAKA,OADF;AAEZE,wBAAU,MAAKA;AAFH,aAAd;AAID;AACF,SAnRkB;;AAEjB,cAAKnC,KAAL,GAAa;AACXD,eAAK,IADM;AAEXkC,mBAASiB,SAFE;AAGXhB,kBAAQgB,SAHG;AAIXf,oBAAUe,SAJC;AAKXC,iBAAO,CALI;AAMXC,kBAAQ;AANG,SAAb;AAFiB;AAUlB;;AAvBkD,8BAyBnDC,kBAzBmD,iCAyB9B;AACnB,aAAKC,QAAL,GAAgB;AACdC,gBAAM,KAAK5C,gBADG;AAEdR,kBAAQ,KAAKiB,gBAFC;AAGd2B,eAAK,KAAKlD,UAHI;AAId2D,kBAAQ,KAAKhD,eAJC;AAKdiD,wBAAcP,SALA;AAMdQ,oBAAU,KAAK3B,gBAND;AAOd4B,sBAAY,KAAKX;AAPH,SAAhB;AASD,OAnCkD;;AAAA,8BAqCnDY,iBArCmD,gCAqC/B;AAClB,aAAKC,SAAL;AACD,OAvCkD;;AAAA,8BAyCnDC,mBAzCmD,gCAyC/BC,SAzC+B,EAyCpBC,SAzCoB,EAyCT;AACxC,YAAI,KAAKhE,KAAL,CAAWD,GAAX,KAAmBiE,UAAUjE,GAAjC,EAAsC;AACpC,cAAI,KAAKC,KAAL,CAAWD,GAAf,EAAoB,KAAKkE,OAAL,CAAa,KAAKjE,KAAL,CAAWD,GAAxB,EAA6B,KAA7B;AACpB,cAAIiE,UAAUjE,GAAd,EAAmB;AACjB,iBAAKmE,QAAL,CAAcF,UAAUjE,GAAxB;AACD,WAFD,MAEO;AACL,iBAAK8D,SAAL,CAAeG,UAAUZ,MAAzB;AACD;AACF,SAPD,MAOO,IAAI,CAACY,UAAUjE,GAAf,EAAoB;AACzB,eAAK8D,SAAL,CAAeG,UAAUZ,MAAzB;AACD,SAFM,MAEA;AAAA,cACCnB,OADD,GACuB+B,SADvB,CACC/B,OADD;AAAA,cACUE,QADV,GACuB6B,SADvB,CACU7B,QADV;;AAEL,cAAI,CAACA,QAAL,EAAe;AACbA,uBAAW,KAAKA,QAAL,IAAiB,qBAA5B;AACD,WAFD,MAEO,IAAI,KAAKA,QAAT,EAAmB;AACxBA,uBAAWA,SAASgC,SAAT,CAAmB,KAAKhC,QAAxB,CAAX;AACD;AACD,cAAMiC,gBAAgBjC,SAASkC,MAAT,CAAgB,KAAK3E,KAArB,EAA4B,CAA5B,CAAtB;AACA,cAAI0E,aAAJ,EAAmB;AACjB,iBAAKjC,QAAL,GAAgB,KAAKmC,mBAAL,CAAyB,KAAKtE,KAAL,CAAWD,GAApC,EAAyCoC,QAAzC,CAAhB;AACD;AACD,cAAI,CAACF,OAAL,EAAc;AACZA,sBAAU,KAAKA,OAAL,IAAgB,qBAA1B;AACD,WAFD,MAEO,IAAI,KAAKA,OAAT,EAAkB;AACvBA,sBAAUA,QAAQkC,SAAR,CAAkB,KAAKlC,OAAvB,CAAV;AACD;AACD,cAAMsC,eAAetC,QAAQoC,MAAR,CAAe,KAAK3E,KAApB,EAA2B,CAA3B,CAArB;AACA,cAAI6E,YAAJ,EAAkB;AAChB,iBAAKtC,OAAL,GAAe,KAAKuC,iBAAL,CAAuBR,UAAUjE,GAAjC,EAAsCkC,OAAtC,CAAf;AACD;AACF;AACF,OAxEkD;;AAAA,8BA0EnDwC,oBA1EmD,mCA0E5B;AACrB,aAAKR,OAAL,CAAa,KAAKjE,KAAL,CAAWD,GAAxB,EAA6B,IAA7B;AACD,OA5EkD;;AAAA,8BA6H7C8D,SA7H6C;AAAA,6FA6HnCa,SA7HmC;AAAA;;AAAA;;AAAA;AAAA;AAAA;AAAA;AAAA,2BA8HvB,KAAK1E,KA9HkB,EA8HzCmD,KA9HyC,UA8HzCA,KA9HyC,EA8HlCC,MA9HkC,UA8HlCA,MA9HkC;;AAAA,wBA+H7CD,QAAQ7D,OA/HqC;AAAA;AAAA;AAAA;;AAgI/CgB,0BAAQC,KAAR,6CAAwDrB,OAAxD;AACA,uBAAK4C,QAAL,CAAc;AACZqB,2BAAO,CADK;AAEZC,4BAAQ;AAFI,mBAAd;AAjI+C;AAAA;;AAAA;AAAA,2BAsIX,KAAK3D,KAtIM,EAsIvCsB,OAtIuC,UAsIvCA,OAtIuC,EAsI9B4D,cAtI8B,UAsI9BA,cAtI8B;;AAAA,wBAuI3C5D,WAAW7B,OAvIgC;AAAA;AAAA;AAAA;;AAwIzC0F,4BAxIyC,GAwI5B7D,OAxI4B;;AAyI7C,sBAAI1B,WAAJ,EAAiBuF,aAAgBA,UAAhB,SAA8BvF,WAA9B;AACjBuF,+BAAgBA,UAAhB,SAA8B1F,OAA9B;AACA,uBAAK2C,KAAL,GAAa8C,eAAeE,kBAAf,CAAkCzF,WAAlC,CAAb;;AA3I6C,uBA4IzC,KAAKyC,KA5IoC;AAAA;AAAA;AAAA;;AAAA,wBA6IvC,KAAKD,QAAL,KAAkB,KAAKC,KA7IgB;AAAA;AAAA;AAAA;;AA8IzC,sBAAI,CAAC6C,aAAatB,MAAd,IAAwB9D,OAA5B,EAAqC;AACnCgB,4BAAQwE,IAAR,CAAa,iDAAb;AACD,mBAFD,MAEO;AACL,yBAAKhD,QAAL,CAAc;AACZ/B,2BAAK,IADO;AAEZqD,8BAAQ,CAACsB,aAAatB,MAAd,IAAwB;AAFpB,qBAAd;AAID;AArJwC;;AAAA;AAwJ3C,uBAAKxB,QAAL,GAAgBsB,SAAhB;;AAxJ2C;AA0JvCnD,qBA1JuC,GA0JjC,IAAIgF,6BAAJ,GACTC,OADS,CACDJ,UADC,EACW;AACnBK,qCAAiB,IADE;AAEnBC,+BAAWC,2BAAkBC,UAFV;AAGnBP,wCAAoB;AAAA,6BAAM,OAAKhD,KAAX;AAAA;AAHD,mBADX,EAMTwD,KANS,EA1JiC;;AAiK7CtF,sBAAIuF,OAAJ,GAAc,KAAK9D,WAAnB;AACA,uBAAKM,QAAL,CAAc;AACZ/B,4BADY;AAEZoD,2BAAOA,QAAQ,CAFH;AAGZC,4BAAQ;AAHI,mBAAd;;AAlK6C;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;;AAAA;AAAA;AAAA;;AAAA;AAAA;;AAAA,8BA2KnDc,QA3KmD,qBA2K1CnE,GA3K0C,EA2KrC;AAAA;;AACZ,YAAIA,GAAJ,EAAS;AACPA,cAAIwF,KAAJ,GACGC,IADH,CACQ,YAAM;AAAA,0BACkB,OAAKxF,KADvB;AAAA,gBACFiC,OADE,WACFA,OADE;AAAA,gBACOC,MADP,WACOA,MADP;;AAEV,gBAAI,CAAC,OAAKD,OAAV,EAAmB,OAAKA,OAAL,GAAeA,WAAW,qBAA1B;AACnB,gBAAI,CAAC,OAAKC,MAAV,EAAkB,OAAKA,MAAL,GAAcA,UAAU,qBAAxB;AAClB,mBAAKJ,QAAL,CAAc;AACZI,sBAAQ,OAAKA,MADD;AAEZD,uBAAS,OAAKA,OAFF;AAGZkB,qBAAO;AAHK,aAAd;AAKD,WAVH,EAWG/C,KAXH,CAWS,UAACC,GAAD,EAAS;AACdC,oBAAQwE,IAAR,0DAAoE5F,OAApE,aAAmFmB,GAAnF;AACAN,gBAAI0F,IAAJ;AACA,mBAAKjE,WAAL,CAAiBnB,GAAjB;AACD,WAfH;AAgBD;AACF,OA9LkD;;AAAA,8BA8MnD4D,OA9MmD,oBA8M3ClE,GA9M2C,EA8MtC2F,KA9MsC,EA8M/B;AAClB,YAAI3F,GAAJ,EAAS;AACP,cAAM4F,WAAW,EAAjB;;AAEA,cAAID,KAAJ,EAAW;AACT;AACA,iBAAKzD,OAAL,GAAeiB,SAAf;AACAyC,qBAASC,IAAT,CAAc,KAAKpF,eAAL,CAAqB,EAArB,CAAd;AACA;AACD,WALD,MAKO,IAAI,CAAC,KAAKyB,OAAV,EAAmB;AACxB,iBAAKA,OAAL,GAAe,KAAKjC,KAAL,CAAWkC,MAA1B;AACD,WAFM,MAEA,IAAI,KAAKlC,KAAL,CAAWkC,MAAf,EAAuB;AAC5B,iBAAKD,OAAL,GAAe,KAAKA,OAAL,CAAakC,SAAb,CAAuB,KAAKnE,KAAL,CAAWkC,MAAlC,CAAf;AACD;;AAEDzB,kBAAQoF,GAAR,CAAYF,QAAZ,EAAsBH,IAAtB,CAA2B,YAAM;AAC/BzF,gBAAI0F,IAAJ;AACD,WAFD;;AAIA,eAAKvD,MAAL,GAAcgB,SAAd;AACA,eAAKpB,QAAL,CAAc;AACZG,qBAAS,KAAKA,OADF;AAEZC,oBAAQ,KAAKA;AAFD,WAAd;AAID;AACF,OAvOkD;;AAAA,8BAkSnDsC,iBAlSmD,8BAkSjCzE,GAlSiC,EAkS5B+F,YAlS4B,EAkSd;AAAA;;AACnC,YAAI7D,UAAU6D,YAAd;AACA,YAAI/F,OAAO+F,YAAX,EAAyB;AAAA,cACf7F,UADe,GACAF,GADA,CACfE,UADe;;AAEvB,cAAIA,cAAcA,WAAWC,eAAX,KAA+B,CAAjD,EAAoD;AAAA,gBAC1CgC,MAD0C,GAC/B,KAAKlC,KAD0B,CAC1CkC,MAD0C;;AAElD,gBAAI,CAAC,KAAKA,MAAV,EAAkB,KAAKA,MAAL,GAAcA,UAAU,qBAAxB;AAClB,gBAAI,KAAKA,MAAL,CAAYmC,MAAZ,CAAmB,KAAK3E,KAAxB,EAA+B,CAA/B,CAAJ,EAAuC;AACrCuC,wBAAUA,QAAQ8D,UAAR,CAAmB,iBAAyB;AAAA,oBAAvBjH,IAAuB;AAAA,oBAAjBkH,WAAiB;;AACpD,oBAAMC,WAAW,OAAK/D,MAAL,CAAYG,KAAZ,CAAkB,CAACvD,IAAD,CAAlB,CAAjB;AACA,oBAAMoH,WAAWD,WACbD,YAAYxD,SAAZ,CAAsB;AAAA,yBAAWyD,SAAS3D,GAAT,CAAaN,OAAb,CAAX;AAAA,iBAAtB,CADa,GAEbgE,WAFJ;AAGA,uBAAO,CAAClH,IAAD,EAAOoH,QAAP,CAAP;AACD,eANS,CAAV;AAOD;AACDjE,oBAAQ8D,UAAR,CAAmB;AAAA,kBAAEjH,IAAF;AAAA,kBAAQoH,QAAR;AAAA,qBAAsBA,SAASC,GAAT,CAAa;AAAA,uBAAWpG,IAAIqG,EAAJ,CAAOtH,IAAP,EAAakD,OAAb,CAAX;AAAA,eAAb,CAAtB;AAAA,aAAnB;AACA,iBAAKE,MAAL,GAAc,KAAKA,MAAL,CAAYiC,SAAZ,CAAsBlC,OAAtB,CAAd;AACA,iBAAKH,QAAL,CAAc;AACZG,uBAASiB,SADG;AAEZhB,sBAAQ,KAAKA;AAFD,aAAd;AAIA,mBAAOgB,SAAP;AACD;AACF;AACD,eAAOjB,OAAP;AACD,OA5TkD;;AAAA,8BA8TnDqC,mBA9TmD,gCA8T/BvE,GA9T+B,EA8T1BoC,QA9T0B,EA8ThB;AACjC,YAAIpC,OAAOoC,QAAX,EAAqB;AACnBA,mBAAS4D,UAAT,CAAoB;AAAA,gBAAEjH,IAAF;AAAA,gBAAQoH,QAAR;AAAA,mBAAsBA,SAASC,GAAT,CAAa;AAAA,qBAAWpG,IAAIsG,GAAJ,CAAQvH,IAAR,EAAckD,OAAd,CAAX;AAAA,aAAb,CAAtB;AAAA,WAApB;AADmB,cAEXE,MAFW,GAEA,KAAKlC,KAFL,CAEXkC,MAFW;;AAGnB,cAAI,CAAC,KAAKA,MAAV,EAAkB,KAAKA,MAAL,GAAcA,UAAU,qBAAxB;AAClB,eAAKA,MAAL,GAAc,KAAKA,MAAL,CAAY6D,UAAZ,CAAuB,iBAAyB;AAAA,gBAAvBjH,IAAuB;AAAA,gBAAjBkH,WAAiB;;AAC5D,gBAAMM,YAAYnE,SAASE,KAAT,CAAe,CAACvD,IAAD,CAAf,CAAlB;AACA,gBAAMoH,WAAWI,YACbN,YAAYxD,SAAZ,CAAsB;AAAA,qBAAW8D,UAAUhE,GAAV,CAAcN,OAAd,CAAX;AAAA,aAAtB,CADa,GAEbgE,WAFJ;AAGA,mBAAO,CAAClH,IAAD,EAAOoH,QAAP,CAAP;AACD,WANa,CAAd;AAOA,eAAKpE,QAAL,CAAc;AACZI,oBAAQ,KAAKA,MADD;AAEZC,sBAAUe;AAFE,WAAd;AAIA,iBAAOA,SAAP;AACD;AACD,eAAOf,QAAP;AACD,OAjVkD;;AAAA,8BAmVnDoE,MAnVmD,qBAmV1C;AAAA;;AAAA,sBACkD,KAAK9G,KADvD;AAAA,YACCsB,OADD,WACCA,OADD;AAAA,YACU4D,cADV,WACUA,cADV;AAAA,YAC6B6B,gBAD7B;;AAEP,YAAMC,mCAAavH,OAAb,IAAuB,KAAKoE,QAA5B,WAAN;AACA,eACE,8BAAC,gBAAD,eACMkD,gBADN,EAEMC,OAFN,EADF;AAMD,OA5VkD;;AAAA;AAAA,MAUzBC,gBAAMC,aAVmB,UAW5C3H,gBAX4C,GAWzBA,gBAXyB;;;AA+VrDQ,kBAAcX,WAAd,sBAA6CF,eAAeK,gBAAf,CAA7C;;AASA,QAAM4H,oBAAoB,SAApBA,iBAAoB,CAAC5G,KAAD,EAAQ6G,MAAR,EAAmB;AAC3C,UAAI,OAAOA,MAAP,KAAkB,UAAtB,EAAkC,OAAOA,OAAO7G,KAAP,CAAP;AAClC,UAAI,OAAO6G,MAAP,KAAkB,QAAtB,EAAgC,OAAOA,MAAP;AAChC,aAAO,EAAP;AACD,KAJD;;AAMA,QAAMC,qBAAqB,SAArBA,kBAAqB;AAAA,aAAa;AACtCnC,wBAAgB,+BAAmB;AACjCE,8BAAoB;AAAA,mBAAM,UAACkC,UAAD,EAAaC,QAAb,EAA0B;AAClD,kBAAMhH,QAAQgH,UAAd;AACA,qBAAOJ,kBAAkB5G,KAAlB,EAAyBZ,WAAzB,CAAP;AACD,aAHmB;AAAA;AADa,SAAnB,EAKb6H,QALa;AADsB,OAAb;AAAA,KAA3B;;AASA,QAAMC,kBAAkB,SAAlBA,eAAkB,CAAClH,KAAD,EAAW;AACjC,UAAMe,UAAU6F,kBAAkB5G,KAAlB,EAAyBb,WAAzB,CAAhB;AACA,aAAO,EAAE4B,gBAAF,EAAP;AACD,KAHD;;AAKA,WAAO,yBAAQmG,eAAR,EAAyBJ,kBAAzB,EAA6CtH,aAA7C,CAAP;AACD,GA7XqB;AAAA,CAAtB;;kBA+XeT,a","file":"inject.js","sourcesContent":["import React from 'react';\nimport PropTypes from 'prop-types';\nimport axios from 'axios';\nimport { bindActionCreators } from 'redux';\nimport { connect } from 'react-redux';\nimport { Map, Set } from 'immutable';\nimport { HubConnectionBuilder, HttpTransportType } from '@aspnet/signalr';\n\nconst getDisplayName = Component => Component.displayName || Component.name || 'Component';\n\nconst injectSignalR = options => (WrappedComponent) => {\n  const {\n    hubName = '',\n    baseAddress = 'http://localhost:5555',\n    accessToken = null,\n    signalrPath = 'signalr',\n    retries = 3,\n  } = options;\n  const { controller = hubName } = options;\n\n  class InjectSignalR extends React.PureComponent {\n    static WrappedComponent = WrappedComponent;\n\n    constructor(props) {\n      super(props);\n      this.state = {\n        hub: null,\n        pending: undefined,\n        active: undefined,\n        moribund: undefined,\n        retry: 0,\n        create: 0,\n      };\n    }\n\n    componentWillMount() {\n      this.hubProxy = {\n        send: this.sendToController,\n        invoke: this.invokeController,\n        add: this.addToGroup,\n        remove: this.removeFromGroup,\n        connectionId: undefined,\n        register: this.registerListener,\n        unregister: this.unregisterListener,\n      };\n    }\n\n    componentDidMount() {\n      this.createHub();\n    }\n\n    componentWillUpdate(nextProps, nextState) {\n      if (this.state.hub !== nextState.hub) {\n        if (this.state.hub) this.stopHub(this.state.hub, false);\n        if (nextState.hub) {\n          this.startHub(nextState.hub);\n        } else {\n          this.createHub(nextState.create);\n        }\n      } else if (!nextState.hub) {\n        this.createHub(nextState.create);\n      } else {\n        let { pending, moribund } = nextState;\n        if (!moribund) {\n          moribund = this.moribund || Map();\n        } else if (this.moribund) {\n          moribund = moribund.mergeDeep(this.moribund);\n        }\n        const moribundCount = moribund.reduce(this.count, 0);\n        if (moribundCount) {\n          this.moribund = this.inactivateListeners(this.state.hub, moribund);\n        }\n        if (!pending) {\n          pending = this.pending || Map();\n        } else if (this.pending) {\n          pending = pending.mergeDeep(this.pending);\n        }\n        const pendingCount = pending.reduce(this.count, 0);\n        if (pendingCount) {\n          this.pending = this.activateListeners(nextState.hub, pending);\n        }\n      }\n    }\n\n    componentWillUnmount() {\n      this.stopHub(this.state.hub, true);\n    }\n\n    count = (c, s) => c + s.count();\n\n    addToGroup = (group) => {\n      const { hub } = this.state;\n      if (hub) {\n        const { connection } = hub;\n        if (connection && connection.connectionState === 1) {\n          hub.invoke('addToGroup', group)\n            .catch((err) => {\n              console.error(`Error: Adding client to group ${group} in ${hubName} failed.\\n\\n${err}`);\n            });\n        }\n      }\n    };\n\n    removeFromGroup = (group) => {\n      const { hub } = this.state;\n      if (hub) {\n        const { connection } = hub;\n        if (connection && connection.connectionState === 1) {\n          return hub.invoke('removeFromGroup', group)\n            .catch((err) => {\n              console.error(`Error: Removing client from group ${group} in ${hubName} failed.\\n\\n${err}`);\n            });\n        }\n      }\n      return Promise.resolve();\n    };\n\n    sendToController = (target, data = null) => {\n      const url = `${this.props.baseUrl}/${controller}/${target}`;\n      const payload = data ? data.toJS() : null;\n      return axios.post(url, payload)\n        .catch((err) => {\n          console.error(`Error: Sending data to ${controller} failed.\\n\\n${err}`);\n        });\n    };\n\n    invokeController = (targetMethod, data = null) => {\n      const urlBase = `${this.props.baseUrl}/${controller}/${targetMethod}`;\n      const url = data ? `${urlBase}/${data}` : urlBase;\n      return axios.get(url)\n        .catch((err) => {\n          console.error(`Error: Invoking ${controller} failed.\\n\\n${err}`);\n        });\n    };\n\n    async createHub(curCreate) {\n      const { retry, create } = this.state;\n      if (retry > retries) {\n        console.error(`Error: Ran out of retries for starting ${hubName}!`);\n        this.setState({\n          retry: 0,\n          create: 0,\n        });\n      } else {\n        const { baseUrl, signalrActions } = this.props;\n        if (baseUrl && hubName) {\n          let hubAddress = baseUrl;\n          if (signalrPath) hubAddress = `${hubAddress}/${signalrPath}`;\n          hubAddress = `${hubAddress}/${hubName}`;\n          this.token = signalrActions.accessTokenFactory(accessToken);\n          if (this.token) {\n            if (this.oldToken === this.token) {\n              if ((curCreate || create) > retries) {\n                console.warn('Warning: Unable to get up-to-date access token.');\n              } else {\n                this.setState({\n                  hub: null,\n                  create: (curCreate || create) + 1,\n                });\n              }\n              return;\n            }\n            this.oldToken = undefined;\n          }\n          const hub = new HubConnectionBuilder()\n            .withUrl(hubAddress, {\n              skipNegotiation: true,\n              transport: HttpTransportType.WebSockets,\n              accessTokenFactory: () => this.token,\n            })\n            .build();\n          hub.onclose = this.handleError;\n          this.setState({\n            hub,\n            retry: retry + 1,\n            create: 0,\n          });\n        }\n      }\n    }\n\n    startHub(hub) {\n      if (hub) {\n        hub.start()\n          .then(() => {\n            const { pending, active } = this.state;\n            if (!this.pending) this.pending = pending || Map();\n            if (!this.active) this.active = active || Map();\n            this.setState({\n              active: this.active,\n              pending: this.pending,\n              retry: 0,\n            });\n          })\n          .catch((err) => {\n            console.warn(`Warning: Error while establishing connection to hub ${hubName}.\\n\\n${err}`);\n            hub.stop();\n            this.handleError(err);\n          });\n      }\n    }\n\n    handleError = (err) => {\n      const { response, statusCode } = err;\n      const { status } = response || {};\n      switch (status || statusCode) {\n        case 500:\n          break;\n        case 401:\n          this.oldToken = this.token; // fall through\n        default:\n          this.setState({ hub: null });\n          break;\n      }\n    };\n\n    stopHub(hub, clear) {\n      if (hub) {\n        const promises = [];\n\n        if (clear) {\n          // Clear pending\n          this.pending = undefined;\n          promises.push(this.removeFromGroup(''));\n          // Merge active to pending\n        } else if (!this.pending) {\n          this.pending = this.state.active;\n        } else if (this.state.active) {\n          this.pending = this.pending.mergeDeep(this.state.active);\n        }\n\n        Promise.all(promises).then(() => {\n          hub.stop();\n        });\n\n        this.active = undefined;\n        this.setState({\n          pending: this.pending,\n          active: this.active,\n        });\n      }\n    }\n\n    registerListener = (name, handler) => {\n      const { pending, active, moribund } = this.state;\n      // Remove listener from moribund listeners\n      if (!this.moribund) this.moribund = moribund || Map();\n      const existingMoribund = this.moribund.getIn([name], Set());\n      if (existingMoribund.has(handler)) {\n        const remainingMoribund = existingMoribund.filterNot(h => h === handler);\n        this.moribund = remainingMoribund.size\n          ? this.moribund.setIn([name], remainingMoribund) : this.moribund.delete(name);\n      }\n      // Add listener to pending listeners (if it is NOT active)\n      if (!this.active) this.active = active || Map();\n      const existingActive = this.active.getIn([name], Set());\n      if (!existingActive.has(handler)) {\n        if (!this.pending) this.pending = pending || Map();\n        const existingPending = this.pending.getIn([name], Set());\n        if (!existingPending.has(handler)) {\n          this.pending = this.pending.setIn([name], existingPending.add(handler));\n        }\n      }\n      if (this.pending !== pending || this.moribund !== moribund) {\n        this.setState({\n          pending: this.pending,\n          moribund: this.moribund,\n        });\n      }\n    };\n\n    unregisterListener = (name, handler) => {\n      const { pending, active, moribund } = this.state;\n      // Remove listener from pending listeners\n      if (!this.pending) this.pending = pending || Map();\n      const existingPending = this.pending.getIn([name], Set());\n      if (existingPending.has(handler)) {\n        const remainingPending = existingPending.filterNot(h => h === handler);\n        this.pending = remainingPending.count()\n          ? this.pending.setIn([name], remainingPending)\n          : this.pending.delete(name);\n      }\n      // Add listener to moribund listeners (if it is active)\n      if (!this.active) this.active = active || Map();\n      const existingActive = this.active.getIn([name], Set());\n      if (existingActive.has(handler)) {\n        if (!this.moribund) this.moribund = moribund || Map();\n        const existingMoribund = this.moribund.getIn([name], Set());\n        if (!existingMoribund.has(handler)) {\n          this.moribund = this.moribund.setIn([name], existingMoribund.add(handler));\n        }\n      }\n      if (this.pending !== pending || this.moribund !== moribund) {\n        this.setState({\n          pending: this.pending,\n          moribund: this.moribund,\n        });\n      }\n    };\n\n    activateListeners(hub, pendingParam) {\n      let pending = pendingParam;\n      if (hub && pendingParam) {\n        const { connection } = hub;\n        if (connection && connection.connectionState === 1) {\n          const { active } = this.state;\n          if (!this.active) this.active = active || Map();\n          if (this.active.reduce(this.count, 0)) {\n            pending = pending.mapEntries(([name, curHandlers]) => {\n              const existing = this.active.getIn([name]);\n              const handlers = existing\n                ? curHandlers.filterNot(handler => existing.has(handler))\n                : curHandlers;\n              return [name, handlers];\n            });\n          }\n          pending.mapEntries(([name, handlers]) => handlers.map(handler => hub.on(name, handler)));\n          this.active = this.active.mergeDeep(pending);\n          this.setState({\n            pending: undefined,\n            active: this.active,\n          });\n          return undefined;\n        }\n      }\n      return pending;\n    }\n\n    inactivateListeners(hub, moribund) {\n      if (hub && moribund) {\n        moribund.mapEntries(([name, handlers]) => handlers.map(handler => hub.off(name, handler)));\n        const { active } = this.state;\n        if (!this.active) this.active = active || Map();\n        this.active = this.active.mapEntries(([name, curHandlers]) => {\n          const removable = moribund.getIn([name]);\n          const handlers = removable\n            ? curHandlers.filterNot(handler => removable.has(handler))\n            : curHandlers;\n          return [name, handlers];\n        });\n        this.setState({\n          active: this.active,\n          moribund: undefined,\n        });\n        return undefined;\n      }\n      return moribund;\n    }\n\n    render() {\n      const { baseUrl, signalrActions, ...passThroughProps } = this.props;\n      const hubProp = { [hubName]: this.hubProxy };\n      return (\n        <WrappedComponent\n          {...passThroughProps}\n          {...hubProp}\n        />\n      );\n    }\n  }\n\n  InjectSignalR.displayName = `InjectSignalR(${getDisplayName(WrappedComponent)})`;\n\n  InjectSignalR.propTypes = {\n    baseUrl: PropTypes.string.isRequired,\n    signalrActions: PropTypes.shape({\n      getAccessToken: PropTypes.func,\n    }).isRequired,\n  };\n\n  const getValueFromState = (state, source) => {\n    if (typeof source === 'function') return source(state);\n    if (typeof source === 'string') return source;\n    return '';\n  };\n\n  const mapDispatchToProps = dispatch => ({\n    signalrActions: bindActionCreators({\n      accessTokenFactory: () => (dispatcher, getState) => {\n        const state = getState();\n        return getValueFromState(state, accessToken);\n      },\n    }, dispatch),\n  });\n\n  const mapStateToProps = (state) => {\n    const baseUrl = getValueFromState(state, baseAddress);\n    return { baseUrl };\n  };\n\n  return connect(mapStateToProps, mapDispatchToProps)(InjectSignalR);\n};\n\nexport default injectSignalR;\n"]} -------------------------------------------------------------------------------- /lib/cjs/types.js: -------------------------------------------------------------------------------- 1 | 'use strict'; 2 | 3 | exports.__esModule = true; 4 | 5 | var _propTypes = require('prop-types'); 6 | 7 | var _propTypes2 = _interopRequireDefault(_propTypes); 8 | 9 | function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } 10 | 11 | var func = _propTypes2.default.func; 12 | 13 | 14 | var hubShape = _propTypes2.default.shape({ 15 | invoke: func.isRequired, 16 | send: func.isRequired, 17 | add: func.isRequired, 18 | remove: func.isRequired, 19 | register: func.isRequired, 20 | unregister: func.isRequired 21 | }); 22 | 23 | exports.default = hubShape; 24 | //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy90eXBlcy5qc3giXSwibmFtZXMiOlsiZnVuYyIsIlByb3BUeXBlcyIsImh1YlNoYXBlIiwic2hhcGUiLCJpbnZva2UiLCJpc1JlcXVpcmVkIiwic2VuZCIsImFkZCIsInJlbW92ZSIsInJlZ2lzdGVyIiwidW5yZWdpc3RlciJdLCJtYXBwaW5ncyI6Ijs7OztBQUFBOzs7Ozs7SUFFUUEsSSxHQUFTQyxtQixDQUFURCxJOzs7QUFFUixJQUFNRSxXQUFXRCxvQkFBVUUsS0FBVixDQUFnQjtBQUMvQkMsVUFBUUosS0FBS0ssVUFEa0I7QUFFL0JDLFFBQU1OLEtBQUtLLFVBRm9CO0FBRy9CRSxPQUFLUCxLQUFLSyxVQUhxQjtBQUkvQkcsVUFBUVIsS0FBS0ssVUFKa0I7QUFLL0JJLFlBQVVULEtBQUtLLFVBTGdCO0FBTS9CSyxjQUFZVixLQUFLSztBQU5jLENBQWhCLENBQWpCOztrQkFTZUgsUSIsImZpbGUiOiJ0eXBlcy5qcyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCBQcm9wVHlwZXMgZnJvbSAncHJvcC10eXBlcyc7XG5cbmNvbnN0IHsgZnVuYyB9ID0gUHJvcFR5cGVzO1xuXG5jb25zdCBodWJTaGFwZSA9IFByb3BUeXBlcy5zaGFwZSh7XG4gIGludm9rZTogZnVuYy5pc1JlcXVpcmVkLFxuICBzZW5kOiBmdW5jLmlzUmVxdWlyZWQsXG4gIGFkZDogZnVuYy5pc1JlcXVpcmVkLFxuICByZW1vdmU6IGZ1bmMuaXNSZXF1aXJlZCxcbiAgcmVnaXN0ZXI6IGZ1bmMuaXNSZXF1aXJlZCxcbiAgdW5yZWdpc3RlcjogZnVuYy5pc1JlcXVpcmVkLFxufSk7XG5cbmV4cG9ydCBkZWZhdWx0IGh1YlNoYXBlO1xuIl19 -------------------------------------------------------------------------------- /lib/es/index.js: -------------------------------------------------------------------------------- 1 | export { default as hubShape } from './types'; 2 | export { default as injectSignalR } from './inject'; 3 | //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9pbmRleC5qcyJdLCJuYW1lcyI6WyJkZWZhdWx0IiwiaHViU2hhcGUiLCJpbmplY3RTaWduYWxSIl0sIm1hcHBpbmdzIjoiQUFBQSxTQUFTQSxXQUFXQyxRQUFwQixRQUFvQyxTQUFwQztBQUNBLFNBQVNELFdBQVdFLGFBQXBCLFFBQXlDLFVBQXpDIiwiZmlsZSI6ImluZGV4LmpzIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0IHsgZGVmYXVsdCBhcyBodWJTaGFwZSB9IGZyb20gJy4vdHlwZXMnO1xuZXhwb3J0IHsgZGVmYXVsdCBhcyBpbmplY3RTaWduYWxSIH0gZnJvbSAnLi9pbmplY3QnO1xuIl19 -------------------------------------------------------------------------------- /lib/es/inject.js: -------------------------------------------------------------------------------- 1 | var _extends = Object.assign || function (target) { for (var i = 1; i < arguments.length; i++) { var source = arguments[i]; for (var key in source) { if (Object.prototype.hasOwnProperty.call(source, key)) { target[key] = source[key]; } } } return target; }; 2 | 3 | function _objectWithoutProperties(obj, keys) { var target = {}; for (var i in obj) { if (keys.indexOf(i) >= 0) continue; if (!Object.prototype.hasOwnProperty.call(obj, i)) continue; target[i] = obj[i]; } return target; } 4 | 5 | function _asyncToGenerator(fn) { return function () { var gen = fn.apply(this, arguments); return new Promise(function (resolve, reject) { function step(key, arg) { try { var info = gen[key](arg); var value = info.value; } catch (error) { reject(error); return; } if (info.done) { resolve(value); } else { return Promise.resolve(value).then(function (value) { step("next", value); }, function (err) { step("throw", err); }); } } return step("next"); }); }; } 6 | 7 | function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } } 8 | 9 | function _possibleConstructorReturn(self, call) { if (!self) { throw new ReferenceError("this hasn't been initialised - super() hasn't been called"); } return call && (typeof call === "object" || typeof call === "function") ? call : self; } 10 | 11 | function _inherits(subClass, superClass) { if (typeof superClass !== "function" && superClass !== null) { throw new TypeError("Super expression must either be null or a function, not " + typeof superClass); } subClass.prototype = Object.create(superClass && superClass.prototype, { constructor: { value: subClass, enumerable: false, writable: true, configurable: true } }); if (superClass) Object.setPrototypeOf ? Object.setPrototypeOf(subClass, superClass) : subClass.__proto__ = superClass; } 12 | 13 | import React from 'react'; 14 | import PropTypes from 'prop-types'; 15 | import axios from 'axios'; 16 | import { bindActionCreators } from 'redux'; 17 | import { connect } from 'react-redux'; 18 | import { Map, Set } from 'immutable'; 19 | import { HubConnectionBuilder, HttpTransportType } from '@aspnet/signalr'; 20 | 21 | var getDisplayName = function getDisplayName(Component) { 22 | return Component.displayName || Component.name || 'Component'; 23 | }; 24 | 25 | var injectSignalR = function injectSignalR(options) { 26 | return function (WrappedComponent) { 27 | var _class, _temp; 28 | 29 | var _options$hubName = options.hubName, 30 | hubName = _options$hubName === undefined ? '' : _options$hubName, 31 | _options$baseAddress = options.baseAddress, 32 | baseAddress = _options$baseAddress === undefined ? 'http://localhost:5555' : _options$baseAddress, 33 | _options$accessToken = options.accessToken, 34 | accessToken = _options$accessToken === undefined ? null : _options$accessToken, 35 | _options$signalrPath = options.signalrPath, 36 | signalrPath = _options$signalrPath === undefined ? 'signalr' : _options$signalrPath, 37 | _options$retries = options.retries, 38 | retries = _options$retries === undefined ? 3 : _options$retries; 39 | var _options$controller = options.controller, 40 | controller = _options$controller === undefined ? hubName : _options$controller; 41 | var InjectSignalR = (_temp = _class = function (_React$PureComponent) { 42 | _inherits(InjectSignalR, _React$PureComponent); 43 | 44 | function InjectSignalR(props) { 45 | _classCallCheck(this, InjectSignalR); 46 | 47 | var _this = _possibleConstructorReturn(this, _React$PureComponent.call(this, props)); 48 | 49 | _this.count = function (c, s) { 50 | return c + s.count(); 51 | }; 52 | 53 | _this.addToGroup = function (group) { 54 | var hub = _this.state.hub; 55 | 56 | if (hub) { 57 | var connection = hub.connection; 58 | 59 | if (connection && connection.connectionState === 1) { 60 | hub.invoke('addToGroup', group).catch(function (err) { 61 | console.error('Error: Adding client to group ' + group + ' in ' + hubName + ' failed.\n\n' + err); 62 | }); 63 | } 64 | } 65 | }; 66 | 67 | _this.removeFromGroup = function (group) { 68 | var hub = _this.state.hub; 69 | 70 | if (hub) { 71 | var connection = hub.connection; 72 | 73 | if (connection && connection.connectionState === 1) { 74 | return hub.invoke('removeFromGroup', group).catch(function (err) { 75 | console.error('Error: Removing client from group ' + group + ' in ' + hubName + ' failed.\n\n' + err); 76 | }); 77 | } 78 | } 79 | return Promise.resolve(); 80 | }; 81 | 82 | _this.sendToController = function (target) { 83 | var data = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : null; 84 | 85 | var url = _this.props.baseUrl + '/' + controller + '/' + target; 86 | var payload = data ? data.toJS() : null; 87 | return axios.post(url, payload).catch(function (err) { 88 | console.error('Error: Sending data to ' + controller + ' failed.\n\n' + err); 89 | }); 90 | }; 91 | 92 | _this.invokeController = function (targetMethod) { 93 | var data = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : null; 94 | 95 | var urlBase = _this.props.baseUrl + '/' + controller + '/' + targetMethod; 96 | var url = data ? urlBase + '/' + data : urlBase; 97 | return axios.get(url).catch(function (err) { 98 | console.error('Error: Invoking ' + controller + ' failed.\n\n' + err); 99 | }); 100 | }; 101 | 102 | _this.handleError = function (err) { 103 | var response = err.response, 104 | statusCode = err.statusCode; 105 | 106 | var _ref = response || {}, 107 | status = _ref.status; 108 | 109 | switch (status || statusCode) { 110 | case 500: 111 | break; 112 | case 401: 113 | _this.oldToken = _this.token; // fall through 114 | default: 115 | _this.setState({ hub: null }); 116 | break; 117 | } 118 | }; 119 | 120 | _this.registerListener = function (name, handler) { 121 | var _this$state = _this.state, 122 | pending = _this$state.pending, 123 | active = _this$state.active, 124 | moribund = _this$state.moribund; 125 | // Remove listener from moribund listeners 126 | 127 | if (!_this.moribund) _this.moribund = moribund || Map(); 128 | var existingMoribund = _this.moribund.getIn([name], Set()); 129 | if (existingMoribund.has(handler)) { 130 | var remainingMoribund = existingMoribund.filterNot(function (h) { 131 | return h === handler; 132 | }); 133 | _this.moribund = remainingMoribund.size ? _this.moribund.setIn([name], remainingMoribund) : _this.moribund.delete(name); 134 | } 135 | // Add listener to pending listeners (if it is NOT active) 136 | if (!_this.active) _this.active = active || Map(); 137 | var existingActive = _this.active.getIn([name], Set()); 138 | if (!existingActive.has(handler)) { 139 | if (!_this.pending) _this.pending = pending || Map(); 140 | var existingPending = _this.pending.getIn([name], Set()); 141 | if (!existingPending.has(handler)) { 142 | _this.pending = _this.pending.setIn([name], existingPending.add(handler)); 143 | } 144 | } 145 | if (_this.pending !== pending || _this.moribund !== moribund) { 146 | _this.setState({ 147 | pending: _this.pending, 148 | moribund: _this.moribund 149 | }); 150 | } 151 | }; 152 | 153 | _this.unregisterListener = function (name, handler) { 154 | var _this$state2 = _this.state, 155 | pending = _this$state2.pending, 156 | active = _this$state2.active, 157 | moribund = _this$state2.moribund; 158 | // Remove listener from pending listeners 159 | 160 | if (!_this.pending) _this.pending = pending || Map(); 161 | var existingPending = _this.pending.getIn([name], Set()); 162 | if (existingPending.has(handler)) { 163 | var remainingPending = existingPending.filterNot(function (h) { 164 | return h === handler; 165 | }); 166 | _this.pending = remainingPending.count() ? _this.pending.setIn([name], remainingPending) : _this.pending.delete(name); 167 | } 168 | // Add listener to moribund listeners (if it is active) 169 | if (!_this.active) _this.active = active || Map(); 170 | var existingActive = _this.active.getIn([name], Set()); 171 | if (existingActive.has(handler)) { 172 | if (!_this.moribund) _this.moribund = moribund || Map(); 173 | var existingMoribund = _this.moribund.getIn([name], Set()); 174 | if (!existingMoribund.has(handler)) { 175 | _this.moribund = _this.moribund.setIn([name], existingMoribund.add(handler)); 176 | } 177 | } 178 | if (_this.pending !== pending || _this.moribund !== moribund) { 179 | _this.setState({ 180 | pending: _this.pending, 181 | moribund: _this.moribund 182 | }); 183 | } 184 | }; 185 | 186 | _this.state = { 187 | hub: null, 188 | pending: undefined, 189 | active: undefined, 190 | moribund: undefined, 191 | retry: 0, 192 | create: 0 193 | }; 194 | return _this; 195 | } 196 | 197 | InjectSignalR.prototype.componentWillMount = function componentWillMount() { 198 | this.hubProxy = { 199 | send: this.sendToController, 200 | invoke: this.invokeController, 201 | add: this.addToGroup, 202 | remove: this.removeFromGroup, 203 | connectionId: undefined, 204 | register: this.registerListener, 205 | unregister: this.unregisterListener 206 | }; 207 | }; 208 | 209 | InjectSignalR.prototype.componentDidMount = function componentDidMount() { 210 | this.createHub(); 211 | }; 212 | 213 | InjectSignalR.prototype.componentWillUpdate = function componentWillUpdate(nextProps, nextState) { 214 | if (this.state.hub !== nextState.hub) { 215 | if (this.state.hub) this.stopHub(this.state.hub, false); 216 | if (nextState.hub) { 217 | this.startHub(nextState.hub); 218 | } else { 219 | this.createHub(nextState.create); 220 | } 221 | } else if (!nextState.hub) { 222 | this.createHub(nextState.create); 223 | } else { 224 | var pending = nextState.pending, 225 | moribund = nextState.moribund; 226 | 227 | if (!moribund) { 228 | moribund = this.moribund || Map(); 229 | } else if (this.moribund) { 230 | moribund = moribund.mergeDeep(this.moribund); 231 | } 232 | var moribundCount = moribund.reduce(this.count, 0); 233 | if (moribundCount) { 234 | this.moribund = this.inactivateListeners(this.state.hub, moribund); 235 | } 236 | if (!pending) { 237 | pending = this.pending || Map(); 238 | } else if (this.pending) { 239 | pending = pending.mergeDeep(this.pending); 240 | } 241 | var pendingCount = pending.reduce(this.count, 0); 242 | if (pendingCount) { 243 | this.pending = this.activateListeners(nextState.hub, pending); 244 | } 245 | } 246 | }; 247 | 248 | InjectSignalR.prototype.componentWillUnmount = function componentWillUnmount() { 249 | this.stopHub(this.state.hub, true); 250 | }; 251 | 252 | InjectSignalR.prototype.createHub = function () { 253 | var _ref2 = _asyncToGenerator( /*#__PURE__*/regeneratorRuntime.mark(function _callee(curCreate) { 254 | var _this2 = this; 255 | 256 | var _state, retry, create, _props, baseUrl, signalrActions, hubAddress, hub; 257 | 258 | return regeneratorRuntime.wrap(function _callee$(_context) { 259 | while (1) { 260 | switch (_context.prev = _context.next) { 261 | case 0: 262 | _state = this.state, retry = _state.retry, create = _state.create; 263 | 264 | if (!(retry > retries)) { 265 | _context.next = 6; 266 | break; 267 | } 268 | 269 | console.error('Error: Ran out of retries for starting ' + hubName + '!'); 270 | this.setState({ 271 | retry: 0, 272 | create: 0 273 | }); 274 | _context.next = 20; 275 | break; 276 | 277 | case 6: 278 | _props = this.props, baseUrl = _props.baseUrl, signalrActions = _props.signalrActions; 279 | 280 | if (!(baseUrl && hubName)) { 281 | _context.next = 20; 282 | break; 283 | } 284 | 285 | hubAddress = baseUrl; 286 | 287 | if (signalrPath) hubAddress = hubAddress + '/' + signalrPath; 288 | hubAddress = hubAddress + '/' + hubName; 289 | this.token = signalrActions.accessTokenFactory(accessToken); 290 | 291 | if (!this.token) { 292 | _context.next = 17; 293 | break; 294 | } 295 | 296 | if (!(this.oldToken === this.token)) { 297 | _context.next = 16; 298 | break; 299 | } 300 | 301 | if ((curCreate || create) > retries) { 302 | console.warn('Warning: Unable to get up-to-date access token.'); 303 | } else { 304 | this.setState({ 305 | hub: null, 306 | create: (curCreate || create) + 1 307 | }); 308 | } 309 | return _context.abrupt('return'); 310 | 311 | case 16: 312 | this.oldToken = undefined; 313 | 314 | case 17: 315 | hub = new HubConnectionBuilder().withUrl(hubAddress, { 316 | skipNegotiation: true, 317 | transport: HttpTransportType.WebSockets, 318 | accessTokenFactory: function accessTokenFactory() { 319 | return _this2.token; 320 | } 321 | }).build(); 322 | 323 | hub.onclose = this.handleError; 324 | this.setState({ 325 | hub: hub, 326 | retry: retry + 1, 327 | create: 0 328 | }); 329 | 330 | case 20: 331 | case 'end': 332 | return _context.stop(); 333 | } 334 | } 335 | }, _callee, this); 336 | })); 337 | 338 | function createHub(_x3) { 339 | return _ref2.apply(this, arguments); 340 | } 341 | 342 | return createHub; 343 | }(); 344 | 345 | InjectSignalR.prototype.startHub = function startHub(hub) { 346 | var _this3 = this; 347 | 348 | if (hub) { 349 | hub.start().then(function () { 350 | var _state2 = _this3.state, 351 | pending = _state2.pending, 352 | active = _state2.active; 353 | 354 | if (!_this3.pending) _this3.pending = pending || Map(); 355 | if (!_this3.active) _this3.active = active || Map(); 356 | _this3.setState({ 357 | active: _this3.active, 358 | pending: _this3.pending, 359 | retry: 0 360 | }); 361 | }).catch(function (err) { 362 | console.warn('Warning: Error while establishing connection to hub ' + hubName + '.\n\n' + err); 363 | hub.stop(); 364 | _this3.handleError(err); 365 | }); 366 | } 367 | }; 368 | 369 | InjectSignalR.prototype.stopHub = function stopHub(hub, clear) { 370 | if (hub) { 371 | var promises = []; 372 | 373 | if (clear) { 374 | // Clear pending 375 | this.pending = undefined; 376 | promises.push(this.removeFromGroup('')); 377 | // Merge active to pending 378 | } else if (!this.pending) { 379 | this.pending = this.state.active; 380 | } else if (this.state.active) { 381 | this.pending = this.pending.mergeDeep(this.state.active); 382 | } 383 | 384 | Promise.all(promises).then(function () { 385 | hub.stop(); 386 | }); 387 | 388 | this.active = undefined; 389 | this.setState({ 390 | pending: this.pending, 391 | active: this.active 392 | }); 393 | } 394 | }; 395 | 396 | InjectSignalR.prototype.activateListeners = function activateListeners(hub, pendingParam) { 397 | var _this4 = this; 398 | 399 | var pending = pendingParam; 400 | if (hub && pendingParam) { 401 | var connection = hub.connection; 402 | 403 | if (connection && connection.connectionState === 1) { 404 | var active = this.state.active; 405 | 406 | if (!this.active) this.active = active || Map(); 407 | if (this.active.reduce(this.count, 0)) { 408 | pending = pending.mapEntries(function (_ref3) { 409 | var name = _ref3[0], 410 | curHandlers = _ref3[1]; 411 | 412 | var existing = _this4.active.getIn([name]); 413 | var handlers = existing ? curHandlers.filterNot(function (handler) { 414 | return existing.has(handler); 415 | }) : curHandlers; 416 | return [name, handlers]; 417 | }); 418 | } 419 | pending.mapEntries(function (_ref4) { 420 | var name = _ref4[0], 421 | handlers = _ref4[1]; 422 | return handlers.map(function (handler) { 423 | return hub.on(name, handler); 424 | }); 425 | }); 426 | this.active = this.active.mergeDeep(pending); 427 | this.setState({ 428 | pending: undefined, 429 | active: this.active 430 | }); 431 | return undefined; 432 | } 433 | } 434 | return pending; 435 | }; 436 | 437 | InjectSignalR.prototype.inactivateListeners = function inactivateListeners(hub, moribund) { 438 | if (hub && moribund) { 439 | moribund.mapEntries(function (_ref5) { 440 | var name = _ref5[0], 441 | handlers = _ref5[1]; 442 | return handlers.map(function (handler) { 443 | return hub.off(name, handler); 444 | }); 445 | }); 446 | var active = this.state.active; 447 | 448 | if (!this.active) this.active = active || Map(); 449 | this.active = this.active.mapEntries(function (_ref6) { 450 | var name = _ref6[0], 451 | curHandlers = _ref6[1]; 452 | 453 | var removable = moribund.getIn([name]); 454 | var handlers = removable ? curHandlers.filterNot(function (handler) { 455 | return removable.has(handler); 456 | }) : curHandlers; 457 | return [name, handlers]; 458 | }); 459 | this.setState({ 460 | active: this.active, 461 | moribund: undefined 462 | }); 463 | return undefined; 464 | } 465 | return moribund; 466 | }; 467 | 468 | InjectSignalR.prototype.render = function render() { 469 | var _hubProp; 470 | 471 | var _props2 = this.props, 472 | baseUrl = _props2.baseUrl, 473 | signalrActions = _props2.signalrActions, 474 | passThroughProps = _objectWithoutProperties(_props2, ['baseUrl', 'signalrActions']); 475 | 476 | var hubProp = (_hubProp = {}, _hubProp[hubName] = this.hubProxy, _hubProp); 477 | return React.createElement(WrappedComponent, _extends({}, passThroughProps, hubProp)); 478 | }; 479 | 480 | return InjectSignalR; 481 | }(React.PureComponent), _class.WrappedComponent = WrappedComponent, _temp); 482 | 483 | 484 | InjectSignalR.displayName = 'InjectSignalR(' + getDisplayName(WrappedComponent) + ')'; 485 | 486 | var getValueFromState = function getValueFromState(state, source) { 487 | if (typeof source === 'function') return source(state); 488 | if (typeof source === 'string') return source; 489 | return ''; 490 | }; 491 | 492 | var mapDispatchToProps = function mapDispatchToProps(dispatch) { 493 | return { 494 | signalrActions: bindActionCreators({ 495 | accessTokenFactory: function accessTokenFactory() { 496 | return function (dispatcher, getState) { 497 | var state = getState(); 498 | return getValueFromState(state, accessToken); 499 | }; 500 | } 501 | }, dispatch) 502 | }; 503 | }; 504 | 505 | var mapStateToProps = function mapStateToProps(state) { 506 | var baseUrl = getValueFromState(state, baseAddress); 507 | return { baseUrl: baseUrl }; 508 | }; 509 | 510 | return connect(mapStateToProps, mapDispatchToProps)(InjectSignalR); 511 | }; 512 | }; 513 | 514 | export default injectSignalR; 515 | //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/inject.jsx"],"names":["React","PropTypes","axios","bindActionCreators","connect","Map","Set","HubConnectionBuilder","HttpTransportType","getDisplayName","Component","displayName","name","injectSignalR","WrappedComponent","options","hubName","baseAddress","accessToken","signalrPath","retries","controller","InjectSignalR","props","count","c","s","addToGroup","group","hub","state","connection","connectionState","invoke","catch","err","console","error","removeFromGroup","Promise","resolve","sendToController","target","data","url","baseUrl","payload","toJS","post","invokeController","targetMethod","urlBase","get","handleError","response","statusCode","status","oldToken","token","setState","registerListener","handler","pending","active","moribund","existingMoribund","getIn","has","remainingMoribund","filterNot","h","size","setIn","delete","existingActive","existingPending","add","unregisterListener","remainingPending","undefined","retry","create","componentWillMount","hubProxy","send","remove","connectionId","register","unregister","componentDidMount","createHub","componentWillUpdate","nextProps","nextState","stopHub","startHub","mergeDeep","moribundCount","reduce","inactivateListeners","pendingCount","activateListeners","componentWillUnmount","curCreate","signalrActions","hubAddress","accessTokenFactory","warn","withUrl","skipNegotiation","transport","WebSockets","build","onclose","start","then","stop","clear","promises","push","all","pendingParam","mapEntries","curHandlers","existing","handlers","map","on","off","removable","render","passThroughProps","hubProp","PureComponent","getValueFromState","source","mapDispatchToProps","dispatcher","getState","dispatch","mapStateToProps"],"mappings":"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","file":"inject.js","sourcesContent":["import React from 'react';\nimport PropTypes from 'prop-types';\nimport axios from 'axios';\nimport { bindActionCreators } from 'redux';\nimport { connect } from 'react-redux';\nimport { Map, Set } from 'immutable';\nimport { HubConnectionBuilder, HttpTransportType } from '@aspnet/signalr';\n\nconst getDisplayName = Component => Component.displayName || Component.name || 'Component';\n\nconst injectSignalR = options => (WrappedComponent) => {\n  const {\n    hubName = '',\n    baseAddress = 'http://localhost:5555',\n    accessToken = null,\n    signalrPath = 'signalr',\n    retries = 3,\n  } = options;\n  const { controller = hubName } = options;\n\n  class InjectSignalR extends React.PureComponent {\n    static WrappedComponent = WrappedComponent;\n\n    constructor(props) {\n      super(props);\n      this.state = {\n        hub: null,\n        pending: undefined,\n        active: undefined,\n        moribund: undefined,\n        retry: 0,\n        create: 0,\n      };\n    }\n\n    componentWillMount() {\n      this.hubProxy = {\n        send: this.sendToController,\n        invoke: this.invokeController,\n        add: this.addToGroup,\n        remove: this.removeFromGroup,\n        connectionId: undefined,\n        register: this.registerListener,\n        unregister: this.unregisterListener,\n      };\n    }\n\n    componentDidMount() {\n      this.createHub();\n    }\n\n    componentWillUpdate(nextProps, nextState) {\n      if (this.state.hub !== nextState.hub) {\n        if (this.state.hub) this.stopHub(this.state.hub, false);\n        if (nextState.hub) {\n          this.startHub(nextState.hub);\n        } else {\n          this.createHub(nextState.create);\n        }\n      } else if (!nextState.hub) {\n        this.createHub(nextState.create);\n      } else {\n        let { pending, moribund } = nextState;\n        if (!moribund) {\n          moribund = this.moribund || Map();\n        } else if (this.moribund) {\n          moribund = moribund.mergeDeep(this.moribund);\n        }\n        const moribundCount = moribund.reduce(this.count, 0);\n        if (moribundCount) {\n          this.moribund = this.inactivateListeners(this.state.hub, moribund);\n        }\n        if (!pending) {\n          pending = this.pending || Map();\n        } else if (this.pending) {\n          pending = pending.mergeDeep(this.pending);\n        }\n        const pendingCount = pending.reduce(this.count, 0);\n        if (pendingCount) {\n          this.pending = this.activateListeners(nextState.hub, pending);\n        }\n      }\n    }\n\n    componentWillUnmount() {\n      this.stopHub(this.state.hub, true);\n    }\n\n    count = (c, s) => c + s.count();\n\n    addToGroup = (group) => {\n      const { hub } = this.state;\n      if (hub) {\n        const { connection } = hub;\n        if (connection && connection.connectionState === 1) {\n          hub.invoke('addToGroup', group)\n            .catch((err) => {\n              console.error(`Error: Adding client to group ${group} in ${hubName} failed.\\n\\n${err}`);\n            });\n        }\n      }\n    };\n\n    removeFromGroup = (group) => {\n      const { hub } = this.state;\n      if (hub) {\n        const { connection } = hub;\n        if (connection && connection.connectionState === 1) {\n          return hub.invoke('removeFromGroup', group)\n            .catch((err) => {\n              console.error(`Error: Removing client from group ${group} in ${hubName} failed.\\n\\n${err}`);\n            });\n        }\n      }\n      return Promise.resolve();\n    };\n\n    sendToController = (target, data = null) => {\n      const url = `${this.props.baseUrl}/${controller}/${target}`;\n      const payload = data ? data.toJS() : null;\n      return axios.post(url, payload)\n        .catch((err) => {\n          console.error(`Error: Sending data to ${controller} failed.\\n\\n${err}`);\n        });\n    };\n\n    invokeController = (targetMethod, data = null) => {\n      const urlBase = `${this.props.baseUrl}/${controller}/${targetMethod}`;\n      const url = data ? `${urlBase}/${data}` : urlBase;\n      return axios.get(url)\n        .catch((err) => {\n          console.error(`Error: Invoking ${controller} failed.\\n\\n${err}`);\n        });\n    };\n\n    async createHub(curCreate) {\n      const { retry, create } = this.state;\n      if (retry > retries) {\n        console.error(`Error: Ran out of retries for starting ${hubName}!`);\n        this.setState({\n          retry: 0,\n          create: 0,\n        });\n      } else {\n        const { baseUrl, signalrActions } = this.props;\n        if (baseUrl && hubName) {\n          let hubAddress = baseUrl;\n          if (signalrPath) hubAddress = `${hubAddress}/${signalrPath}`;\n          hubAddress = `${hubAddress}/${hubName}`;\n          this.token = signalrActions.accessTokenFactory(accessToken);\n          if (this.token) {\n            if (this.oldToken === this.token) {\n              if ((curCreate || create) > retries) {\n                console.warn('Warning: Unable to get up-to-date access token.');\n              } else {\n                this.setState({\n                  hub: null,\n                  create: (curCreate || create) + 1,\n                });\n              }\n              return;\n            }\n            this.oldToken = undefined;\n          }\n          const hub = new HubConnectionBuilder()\n            .withUrl(hubAddress, {\n              skipNegotiation: true,\n              transport: HttpTransportType.WebSockets,\n              accessTokenFactory: () => this.token,\n            })\n            .build();\n          hub.onclose = this.handleError;\n          this.setState({\n            hub,\n            retry: retry + 1,\n            create: 0,\n          });\n        }\n      }\n    }\n\n    startHub(hub) {\n      if (hub) {\n        hub.start()\n          .then(() => {\n            const { pending, active } = this.state;\n            if (!this.pending) this.pending = pending || Map();\n            if (!this.active) this.active = active || Map();\n            this.setState({\n              active: this.active,\n              pending: this.pending,\n              retry: 0,\n            });\n          })\n          .catch((err) => {\n            console.warn(`Warning: Error while establishing connection to hub ${hubName}.\\n\\n${err}`);\n            hub.stop();\n            this.handleError(err);\n          });\n      }\n    }\n\n    handleError = (err) => {\n      const { response, statusCode } = err;\n      const { status } = response || {};\n      switch (status || statusCode) {\n        case 500:\n          break;\n        case 401:\n          this.oldToken = this.token; // fall through\n        default:\n          this.setState({ hub: null });\n          break;\n      }\n    };\n\n    stopHub(hub, clear) {\n      if (hub) {\n        const promises = [];\n\n        if (clear) {\n          // Clear pending\n          this.pending = undefined;\n          promises.push(this.removeFromGroup(''));\n          // Merge active to pending\n        } else if (!this.pending) {\n          this.pending = this.state.active;\n        } else if (this.state.active) {\n          this.pending = this.pending.mergeDeep(this.state.active);\n        }\n\n        Promise.all(promises).then(() => {\n          hub.stop();\n        });\n\n        this.active = undefined;\n        this.setState({\n          pending: this.pending,\n          active: this.active,\n        });\n      }\n    }\n\n    registerListener = (name, handler) => {\n      const { pending, active, moribund } = this.state;\n      // Remove listener from moribund listeners\n      if (!this.moribund) this.moribund = moribund || Map();\n      const existingMoribund = this.moribund.getIn([name], Set());\n      if (existingMoribund.has(handler)) {\n        const remainingMoribund = existingMoribund.filterNot(h => h === handler);\n        this.moribund = remainingMoribund.size\n          ? this.moribund.setIn([name], remainingMoribund) : this.moribund.delete(name);\n      }\n      // Add listener to pending listeners (if it is NOT active)\n      if (!this.active) this.active = active || Map();\n      const existingActive = this.active.getIn([name], Set());\n      if (!existingActive.has(handler)) {\n        if (!this.pending) this.pending = pending || Map();\n        const existingPending = this.pending.getIn([name], Set());\n        if (!existingPending.has(handler)) {\n          this.pending = this.pending.setIn([name], existingPending.add(handler));\n        }\n      }\n      if (this.pending !== pending || this.moribund !== moribund) {\n        this.setState({\n          pending: this.pending,\n          moribund: this.moribund,\n        });\n      }\n    };\n\n    unregisterListener = (name, handler) => {\n      const { pending, active, moribund } = this.state;\n      // Remove listener from pending listeners\n      if (!this.pending) this.pending = pending || Map();\n      const existingPending = this.pending.getIn([name], Set());\n      if (existingPending.has(handler)) {\n        const remainingPending = existingPending.filterNot(h => h === handler);\n        this.pending = remainingPending.count()\n          ? this.pending.setIn([name], remainingPending)\n          : this.pending.delete(name);\n      }\n      // Add listener to moribund listeners (if it is active)\n      if (!this.active) this.active = active || Map();\n      const existingActive = this.active.getIn([name], Set());\n      if (existingActive.has(handler)) {\n        if (!this.moribund) this.moribund = moribund || Map();\n        const existingMoribund = this.moribund.getIn([name], Set());\n        if (!existingMoribund.has(handler)) {\n          this.moribund = this.moribund.setIn([name], existingMoribund.add(handler));\n        }\n      }\n      if (this.pending !== pending || this.moribund !== moribund) {\n        this.setState({\n          pending: this.pending,\n          moribund: this.moribund,\n        });\n      }\n    };\n\n    activateListeners(hub, pendingParam) {\n      let pending = pendingParam;\n      if (hub && pendingParam) {\n        const { connection } = hub;\n        if (connection && connection.connectionState === 1) {\n          const { active } = this.state;\n          if (!this.active) this.active = active || Map();\n          if (this.active.reduce(this.count, 0)) {\n            pending = pending.mapEntries(([name, curHandlers]) => {\n              const existing = this.active.getIn([name]);\n              const handlers = existing\n                ? curHandlers.filterNot(handler => existing.has(handler))\n                : curHandlers;\n              return [name, handlers];\n            });\n          }\n          pending.mapEntries(([name, handlers]) => handlers.map(handler => hub.on(name, handler)));\n          this.active = this.active.mergeDeep(pending);\n          this.setState({\n            pending: undefined,\n            active: this.active,\n          });\n          return undefined;\n        }\n      }\n      return pending;\n    }\n\n    inactivateListeners(hub, moribund) {\n      if (hub && moribund) {\n        moribund.mapEntries(([name, handlers]) => handlers.map(handler => hub.off(name, handler)));\n        const { active } = this.state;\n        if (!this.active) this.active = active || Map();\n        this.active = this.active.mapEntries(([name, curHandlers]) => {\n          const removable = moribund.getIn([name]);\n          const handlers = removable\n            ? curHandlers.filterNot(handler => removable.has(handler))\n            : curHandlers;\n          return [name, handlers];\n        });\n        this.setState({\n          active: this.active,\n          moribund: undefined,\n        });\n        return undefined;\n      }\n      return moribund;\n    }\n\n    render() {\n      const { baseUrl, signalrActions, ...passThroughProps } = this.props;\n      const hubProp = { [hubName]: this.hubProxy };\n      return (\n        <WrappedComponent\n          {...passThroughProps}\n          {...hubProp}\n        />\n      );\n    }\n  }\n\n  InjectSignalR.displayName = `InjectSignalR(${getDisplayName(WrappedComponent)})`;\n\n  InjectSignalR.propTypes = {\n    baseUrl: PropTypes.string.isRequired,\n    signalrActions: PropTypes.shape({\n      getAccessToken: PropTypes.func,\n    }).isRequired,\n  };\n\n  const getValueFromState = (state, source) => {\n    if (typeof source === 'function') return source(state);\n    if (typeof source === 'string') return source;\n    return '';\n  };\n\n  const mapDispatchToProps = dispatch => ({\n    signalrActions: bindActionCreators({\n      accessTokenFactory: () => (dispatcher, getState) => {\n        const state = getState();\n        return getValueFromState(state, accessToken);\n      },\n    }, dispatch),\n  });\n\n  const mapStateToProps = (state) => {\n    const baseUrl = getValueFromState(state, baseAddress);\n    return { baseUrl };\n  };\n\n  return connect(mapStateToProps, mapDispatchToProps)(InjectSignalR);\n};\n\nexport default injectSignalR;\n"]} -------------------------------------------------------------------------------- /lib/es/types.js: -------------------------------------------------------------------------------- 1 | import PropTypes from 'prop-types'; 2 | 3 | var func = PropTypes.func; 4 | 5 | 6 | var hubShape = PropTypes.shape({ 7 | invoke: func.isRequired, 8 | send: func.isRequired, 9 | add: func.isRequired, 10 | remove: func.isRequired, 11 | register: func.isRequired, 12 | unregister: func.isRequired 13 | }); 14 | 15 | export default hubShape; 16 | //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy90eXBlcy5qc3giXSwibmFtZXMiOlsiUHJvcFR5cGVzIiwiZnVuYyIsImh1YlNoYXBlIiwic2hhcGUiLCJpbnZva2UiLCJpc1JlcXVpcmVkIiwic2VuZCIsImFkZCIsInJlbW92ZSIsInJlZ2lzdGVyIiwidW5yZWdpc3RlciJdLCJtYXBwaW5ncyI6IkFBQUEsT0FBT0EsU0FBUCxNQUFzQixZQUF0Qjs7SUFFUUMsSSxHQUFTRCxTLENBQVRDLEk7OztBQUVSLElBQU1DLFdBQVdGLFVBQVVHLEtBQVYsQ0FBZ0I7QUFDL0JDLFVBQVFILEtBQUtJLFVBRGtCO0FBRS9CQyxRQUFNTCxLQUFLSSxVQUZvQjtBQUcvQkUsT0FBS04sS0FBS0ksVUFIcUI7QUFJL0JHLFVBQVFQLEtBQUtJLFVBSmtCO0FBSy9CSSxZQUFVUixLQUFLSSxVQUxnQjtBQU0vQkssY0FBWVQsS0FBS0k7QUFOYyxDQUFoQixDQUFqQjs7QUFTQSxlQUFlSCxRQUFmIiwiZmlsZSI6InR5cGVzLmpzIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IFByb3BUeXBlcyBmcm9tICdwcm9wLXR5cGVzJztcblxuY29uc3QgeyBmdW5jIH0gPSBQcm9wVHlwZXM7XG5cbmNvbnN0IGh1YlNoYXBlID0gUHJvcFR5cGVzLnNoYXBlKHtcbiAgaW52b2tlOiBmdW5jLmlzUmVxdWlyZWQsXG4gIHNlbmQ6IGZ1bmMuaXNSZXF1aXJlZCxcbiAgYWRkOiBmdW5jLmlzUmVxdWlyZWQsXG4gIHJlbW92ZTogZnVuYy5pc1JlcXVpcmVkLFxuICByZWdpc3RlcjogZnVuYy5pc1JlcXVpcmVkLFxuICB1bnJlZ2lzdGVyOiBmdW5jLmlzUmVxdWlyZWQsXG59KTtcblxuZXhwb3J0IGRlZmF1bHQgaHViU2hhcGU7XG4iXX0= -------------------------------------------------------------------------------- /package.json: -------------------------------------------------------------------------------- 1 | { 2 | "author": "OpusCapita (www.opuscapita.com)", 3 | "name": "@opuscapita/react-signalr", 4 | "version": "3.3.0", 5 | "private": false, 6 | "license": "MIT", 7 | "description": "OpusCapita react signalr", 8 | "main": "lib/umd/index.js", 9 | "cjs": "lib/cjs/index.js", 10 | "es": "lib/es/index.js", 11 | "scripts": { 12 | "clean": "rimraf lib", 13 | "build": "npm-run-all clean build:*", 14 | "build:cjs": "cross-env NODE_ENV=production BUILD_ENV=cjs babel src --out-dir lib/cjs --copy-files --source-maps inline", 15 | "build:es": "cross-env NODE_ENV=production BUILD_ENV=es babel src --out-dir lib/es --copy-files --source-maps inline", 16 | "build:umd": "cross-env NODE_ENV=development BUILD_ENV=umd webpack", 17 | "build:umd-min": "cross-env NODE_ENV=production BUILD_ENV=umd webpack", 18 | "watch": "npm-run-all --parallel watch:*", 19 | "watch:cjs": "cross-env NODE_ENV=development BUILD_ENV=cjs babel ./src/index.js --out-dir lib/cjs --copy-files --watch", 20 | "watch:es": "cross-env NODE_ENV=development BUILD_ENV=es babel ./src/index.js --out-dir lib/es --copy-files --watch", 21 | "watch:umd": "cross-env NODE_ENV=development BUILD_ENV=umd webpack --progress --colors --watch", 22 | "lint": "node_modules/.bin/eslint --ext .jsx,.js src/", 23 | "preversion": "npm run lint", 24 | "version": "npm run build && git add -A lib", 25 | "postversion": "git push && git push --tags" 26 | }, 27 | "files": [ 28 | "/lib" 29 | ], 30 | "engines": { 31 | "node": ">=6.10.0", 32 | "npm": ">=5.4.0" 33 | }, 34 | "peerDependencies": { 35 | "prop-types": "15", 36 | "react": "15 || 16", 37 | "react-dom": "15 || 16" 38 | }, 39 | "devDependencies": { 40 | "autoprefixer": "9.4.6", 41 | "babel-cli": "6.26.0", 42 | "babel-eslint": "10.0.1", 43 | "babel-loader": "7.1.5", 44 | "babel-plugin-dynamic-import-node": "2.2.0", 45 | "babel-plugin-react-transform": "3.0.0", 46 | "babel-plugin-transform-decorators-legacy": "1.3.5", 47 | "babel-plugin-transform-react-remove-prop-types": "0.4.23", 48 | "babel-polyfill": "6.26.0", 49 | "babel-preset-env": "1.7.0", 50 | "babel-preset-react": "6.24.1", 51 | "babel-preset-stage-1": "6.24.1", 52 | "babel-register": "6.26.0", 53 | "bootstrap-sass": "3.4.0", 54 | "clean-webpack-plugin": "1.0.1", 55 | "cross-env": "5.2.0", 56 | "css-loader": "0.28.11", 57 | "enzyme": "3.8.0", 58 | "enzyme-adapter-react-16": "1.7.1", 59 | "eslint": "5.12.1", 60 | "eslint-config-airbnb": "16.1.0", 61 | "eslint-plugin-import": "2.15.0", 62 | "eslint-plugin-jsx-a11y": "6.1.2", 63 | "eslint-plugin-react": "7.12.4", 64 | "file-loader": "3.0.1", 65 | "global-jsdom": "4.2.0", 66 | "html-webpack-plugin": "2.30.1", 67 | "ignore-styles": "5.0.1", 68 | "immutable": "3.8.2", 69 | "jsdom": "13.1.0", 70 | "mocha": "5.2.0", 71 | "node-sass": "4.11.0", 72 | "npm-run-all": "4.1.5", 73 | "postcss-flexbugs-fixes": "4.1.0", 74 | "postcss-loader": "2.1.6", 75 | "precss": "4.0.0", 76 | "progress-bar-webpack-plugin": "1.11.0", 77 | "prop-types": "15.6.2", 78 | "react": "16.7.0", 79 | "react-bootstrap": "0.32.4", 80 | "react-dom": "16.7.0", 81 | "react-hot-loader": "4.3.11", 82 | "react-router": "4.3.1", 83 | "react-router-dom": "4.3.1", 84 | "react-svg-loader": "2.1.0", 85 | "react-test-renderer": "16.7.0", 86 | "rimraf": "2.6.3", 87 | "sass-loader": "7.1.0", 88 | "sinon": "7.2.3", 89 | "style-loader": "0.23.1", 90 | "url-loader": "1.1.2", 91 | "webpack": "3.12.0", 92 | "webpack-dev-server": "2.11.3", 93 | "webpack-merge": "4.2.1", 94 | "webpack-node-externals": "1.7.2", 95 | "webpack-notifier": "1.7.0", 96 | "write-file-webpack-plugin": "4.5.0" 97 | }, 98 | "dependencies": { 99 | "@aspnet/signalr": "1.0.0", 100 | "axios": "0.18.0", 101 | "react-redux": "5.0.7", 102 | "redux": "4.0.1", 103 | "tslib": "1.9.3" 104 | }, 105 | "repository": { 106 | "type": "git", 107 | "url": "https://github.com/OpusCapita/react-signalr.git" 108 | } 109 | } 110 | -------------------------------------------------------------------------------- /src/index.js: -------------------------------------------------------------------------------- 1 | export { default as hubShape } from './types'; 2 | export { default as injectSignalR } from './inject'; 3 | -------------------------------------------------------------------------------- /src/inject.jsx: -------------------------------------------------------------------------------- 1 | import React from 'react'; 2 | import PropTypes from 'prop-types'; 3 | import axios from 'axios'; 4 | import { bindActionCreators } from 'redux'; 5 | import { connect } from 'react-redux'; 6 | import { Map, Set } from 'immutable'; 7 | import { HubConnectionBuilder, HttpTransportType } from '@aspnet/signalr'; 8 | 9 | const getDisplayName = Component => Component.displayName || Component.name || 'Component'; 10 | 11 | const injectSignalR = options => (WrappedComponent) => { 12 | const { 13 | hubName = '', 14 | baseAddress = 'http://localhost:5555', 15 | accessToken = null, 16 | signalrPath = 'signalr', 17 | retries = 3, 18 | } = options; 19 | const { controller = hubName } = options; 20 | 21 | class InjectSignalR extends React.PureComponent { 22 | static WrappedComponent = WrappedComponent; 23 | 24 | constructor(props) { 25 | super(props); 26 | this.state = { 27 | hub: null, 28 | pending: undefined, 29 | active: undefined, 30 | moribund: undefined, 31 | retry: 0, 32 | create: 0, 33 | }; 34 | } 35 | 36 | componentWillMount() { 37 | this.hubProxy = { 38 | send: this.sendToController, 39 | invoke: this.invokeController, 40 | add: this.addToGroup, 41 | remove: this.removeFromGroup, 42 | connectionId: undefined, 43 | register: this.registerListener, 44 | unregister: this.unregisterListener, 45 | }; 46 | } 47 | 48 | componentDidMount() { 49 | this.createHub(); 50 | } 51 | 52 | componentWillUpdate(nextProps, nextState) { 53 | if (this.state.hub !== nextState.hub) { 54 | if (this.state.hub) this.stopHub(this.state.hub, false); 55 | if (nextState.hub) { 56 | this.startHub(nextState.hub); 57 | } else { 58 | this.createHub(nextState.create); 59 | } 60 | } else if (!nextState.hub) { 61 | this.createHub(nextState.create); 62 | } else { 63 | let { pending, moribund } = nextState; 64 | if (!moribund) { 65 | moribund = this.moribund || Map(); 66 | } else if (this.moribund) { 67 | moribund = moribund.mergeDeep(this.moribund); 68 | } 69 | const moribundCount = moribund.reduce(this.count, 0); 70 | if (moribundCount) { 71 | this.moribund = this.inactivateListeners(this.state.hub, moribund); 72 | } 73 | if (!pending) { 74 | pending = this.pending || Map(); 75 | } else if (this.pending) { 76 | pending = pending.mergeDeep(this.pending); 77 | } 78 | const pendingCount = pending.reduce(this.count, 0); 79 | if (pendingCount) { 80 | this.pending = this.activateListeners(nextState.hub, pending); 81 | } 82 | } 83 | } 84 | 85 | componentWillUnmount() { 86 | this.stopHub(this.state.hub, true); 87 | } 88 | 89 | count = (c, s) => c + s.count(); 90 | 91 | addToGroup = (group) => { 92 | const { hub } = this.state; 93 | if (hub) { 94 | const { connection } = hub; 95 | if (connection && connection.connectionState === 1) { 96 | hub.invoke('addToGroup', group) 97 | .catch((err) => { 98 | console.error(`Error: Adding client to group ${group} in ${hubName} failed.\n\n${err}`); 99 | }); 100 | } 101 | } 102 | }; 103 | 104 | removeFromGroup = (group) => { 105 | const { hub } = this.state; 106 | if (hub) { 107 | const { connection } = hub; 108 | if (connection && connection.connectionState === 1) { 109 | return hub.invoke('removeFromGroup', group) 110 | .catch((err) => { 111 | console.error(`Error: Removing client from group ${group} in ${hubName} failed.\n\n${err}`); 112 | }); 113 | } 114 | } 115 | return Promise.resolve(); 116 | }; 117 | 118 | sendToController = (target, data = null) => { 119 | const url = `${this.props.baseUrl}/${controller}/${target}`; 120 | const payload = data ? data.toJS() : null; 121 | return axios.post(url, payload) 122 | .catch((err) => { 123 | console.error(`Error: Sending data to ${controller} failed.\n\n${err}`); 124 | }); 125 | }; 126 | 127 | invokeController = (targetMethod, data = null) => { 128 | const urlBase = `${this.props.baseUrl}/${controller}/${targetMethod}`; 129 | const url = data ? `${urlBase}/${data}` : urlBase; 130 | return axios.get(url) 131 | .catch((err) => { 132 | console.error(`Error: Invoking ${controller} failed.\n\n${err}`); 133 | }); 134 | }; 135 | 136 | async createHub(curCreate) { 137 | const { retry, create } = this.state; 138 | if (retry > retries) { 139 | console.error(`Error: Ran out of retries for starting ${hubName}!`); 140 | this.setState({ 141 | retry: 0, 142 | create: 0, 143 | }); 144 | } else { 145 | const { baseUrl, signalrActions } = this.props; 146 | if (baseUrl && hubName) { 147 | let hubAddress = baseUrl; 148 | if (signalrPath) hubAddress = `${hubAddress}/${signalrPath}`; 149 | hubAddress = `${hubAddress}/${hubName}`; 150 | this.token = signalrActions.accessTokenFactory(accessToken); 151 | if (this.token) { 152 | if (this.oldToken === this.token) { 153 | if ((curCreate || create) > retries) { 154 | console.warn('Warning: Unable to get up-to-date access token.'); 155 | } else { 156 | this.setState({ 157 | hub: null, 158 | create: (curCreate || create) + 1, 159 | }); 160 | } 161 | return; 162 | } 163 | this.oldToken = undefined; 164 | } 165 | const hub = new HubConnectionBuilder() 166 | .withUrl(hubAddress, { 167 | skipNegotiation: true, 168 | transport: HttpTransportType.WebSockets, 169 | accessTokenFactory: () => this.token, 170 | }) 171 | .build(); 172 | hub.onclose = this.handleError; 173 | this.setState({ 174 | hub, 175 | retry: retry + 1, 176 | create: 0, 177 | }); 178 | } 179 | } 180 | } 181 | 182 | startHub(hub) { 183 | if (hub) { 184 | hub.start() 185 | .then(() => { 186 | const { pending, active } = this.state; 187 | if (!this.pending) this.pending = pending || Map(); 188 | if (!this.active) this.active = active || Map(); 189 | this.setState({ 190 | active: this.active, 191 | pending: this.pending, 192 | retry: 0, 193 | }); 194 | }) 195 | .catch((err) => { 196 | console.warn(`Warning: Error while establishing connection to hub ${hubName}.\n\n${err}`); 197 | hub.stop(); 198 | this.handleError(err); 199 | }); 200 | } 201 | } 202 | 203 | handleError = (err) => { 204 | const { response, statusCode } = err; 205 | const { status } = response || {}; 206 | switch (status || statusCode) { 207 | case 500: 208 | break; 209 | case 401: 210 | this.oldToken = this.token; // fall through 211 | default: 212 | this.setState({ hub: null }); 213 | break; 214 | } 215 | }; 216 | 217 | stopHub(hub, clear) { 218 | if (hub) { 219 | const promises = []; 220 | 221 | if (clear) { 222 | // Clear pending 223 | this.pending = undefined; 224 | promises.push(this.removeFromGroup('')); 225 | // Merge active to pending 226 | } else if (!this.pending) { 227 | this.pending = this.state.active; 228 | } else if (this.state.active) { 229 | this.pending = this.pending.mergeDeep(this.state.active); 230 | } 231 | 232 | Promise.all(promises).then(() => { 233 | hub.stop(); 234 | }); 235 | 236 | this.active = undefined; 237 | this.setState({ 238 | pending: this.pending, 239 | active: this.active, 240 | }); 241 | } 242 | } 243 | 244 | registerListener = (name, handler) => { 245 | const { pending, active, moribund } = this.state; 246 | // Remove listener from moribund listeners 247 | if (!this.moribund) this.moribund = moribund || Map(); 248 | const existingMoribund = this.moribund.getIn([name], Set()); 249 | if (existingMoribund.has(handler)) { 250 | const remainingMoribund = existingMoribund.filterNot(h => h === handler); 251 | this.moribund = remainingMoribund.size 252 | ? this.moribund.setIn([name], remainingMoribund) : this.moribund.delete(name); 253 | } 254 | // Add listener to pending listeners (if it is NOT active) 255 | if (!this.active) this.active = active || Map(); 256 | const existingActive = this.active.getIn([name], Set()); 257 | if (!existingActive.has(handler)) { 258 | if (!this.pending) this.pending = pending || Map(); 259 | const existingPending = this.pending.getIn([name], Set()); 260 | if (!existingPending.has(handler)) { 261 | this.pending = this.pending.setIn([name], existingPending.add(handler)); 262 | } 263 | } 264 | if (this.pending !== pending || this.moribund !== moribund) { 265 | this.setState({ 266 | pending: this.pending, 267 | moribund: this.moribund, 268 | }); 269 | } 270 | }; 271 | 272 | unregisterListener = (name, handler) => { 273 | const { pending, active, moribund } = this.state; 274 | // Remove listener from pending listeners 275 | if (!this.pending) this.pending = pending || Map(); 276 | const existingPending = this.pending.getIn([name], Set()); 277 | if (existingPending.has(handler)) { 278 | const remainingPending = existingPending.filterNot(h => h === handler); 279 | this.pending = remainingPending.count() 280 | ? this.pending.setIn([name], remainingPending) 281 | : this.pending.delete(name); 282 | } 283 | // Add listener to moribund listeners (if it is active) 284 | if (!this.active) this.active = active || Map(); 285 | const existingActive = this.active.getIn([name], Set()); 286 | if (existingActive.has(handler)) { 287 | if (!this.moribund) this.moribund = moribund || Map(); 288 | const existingMoribund = this.moribund.getIn([name], Set()); 289 | if (!existingMoribund.has(handler)) { 290 | this.moribund = this.moribund.setIn([name], existingMoribund.add(handler)); 291 | } 292 | } 293 | if (this.pending !== pending || this.moribund !== moribund) { 294 | this.setState({ 295 | pending: this.pending, 296 | moribund: this.moribund, 297 | }); 298 | } 299 | }; 300 | 301 | activateListeners(hub, pendingParam) { 302 | let pending = pendingParam; 303 | if (hub && pendingParam) { 304 | const { connection } = hub; 305 | if (connection && connection.connectionState === 1) { 306 | const { active } = this.state; 307 | if (!this.active) this.active = active || Map(); 308 | if (this.active.reduce(this.count, 0)) { 309 | pending = pending.mapEntries(([name, curHandlers]) => { 310 | const existing = this.active.getIn([name]); 311 | const handlers = existing 312 | ? curHandlers.filterNot(handler => existing.has(handler)) 313 | : curHandlers; 314 | return [name, handlers]; 315 | }); 316 | } 317 | pending.mapEntries(([name, handlers]) => handlers.map(handler => hub.on(name, handler))); 318 | this.active = this.active.mergeDeep(pending); 319 | this.setState({ 320 | pending: undefined, 321 | active: this.active, 322 | }); 323 | return undefined; 324 | } 325 | } 326 | return pending; 327 | } 328 | 329 | inactivateListeners(hub, moribund) { 330 | if (hub && moribund) { 331 | moribund.mapEntries(([name, handlers]) => handlers.map(handler => hub.off(name, handler))); 332 | const { active } = this.state; 333 | if (!this.active) this.active = active || Map(); 334 | this.active = this.active.mapEntries(([name, curHandlers]) => { 335 | const removable = moribund.getIn([name]); 336 | const handlers = removable 337 | ? curHandlers.filterNot(handler => removable.has(handler)) 338 | : curHandlers; 339 | return [name, handlers]; 340 | }); 341 | this.setState({ 342 | active: this.active, 343 | moribund: undefined, 344 | }); 345 | return undefined; 346 | } 347 | return moribund; 348 | } 349 | 350 | render() { 351 | const { baseUrl, signalrActions, ...passThroughProps } = this.props; 352 | const hubProp = { [hubName]: this.hubProxy }; 353 | return ( 354 | 358 | ); 359 | } 360 | } 361 | 362 | InjectSignalR.displayName = `InjectSignalR(${getDisplayName(WrappedComponent)})`; 363 | 364 | InjectSignalR.propTypes = { 365 | baseUrl: PropTypes.string.isRequired, 366 | signalrActions: PropTypes.shape({ 367 | getAccessToken: PropTypes.func, 368 | }).isRequired, 369 | }; 370 | 371 | const getValueFromState = (state, source) => { 372 | if (typeof source === 'function') return source(state); 373 | if (typeof source === 'string') return source; 374 | return ''; 375 | }; 376 | 377 | const mapDispatchToProps = dispatch => ({ 378 | signalrActions: bindActionCreators({ 379 | accessTokenFactory: () => (dispatcher, getState) => { 380 | const state = getState(); 381 | return getValueFromState(state, accessToken); 382 | }, 383 | }, dispatch), 384 | }); 385 | 386 | const mapStateToProps = (state) => { 387 | const baseUrl = getValueFromState(state, baseAddress); 388 | return { baseUrl }; 389 | }; 390 | 391 | return connect(mapStateToProps, mapDispatchToProps)(InjectSignalR); 392 | }; 393 | 394 | export default injectSignalR; 395 | -------------------------------------------------------------------------------- /src/types.jsx: -------------------------------------------------------------------------------- 1 | import PropTypes from 'prop-types'; 2 | 3 | const { func } = PropTypes; 4 | 5 | const hubShape = PropTypes.shape({ 6 | invoke: func.isRequired, 7 | send: func.isRequired, 8 | add: func.isRequired, 9 | remove: func.isRequired, 10 | register: func.isRequired, 11 | unregister: func.isRequired, 12 | }); 13 | 14 | export default hubShape; 15 | -------------------------------------------------------------------------------- /tools/babel.preset.js: -------------------------------------------------------------------------------- 1 | const { NODE_ENV, BUILD_ENV } = process.env; 2 | const presetOptions = BUILD_ENV === 'hot' || BUILD_ENV === 'umd' || BUILD_ENV === 'es' ? 3 | { loose: true, modules: false } : 4 | { loose: true }; 5 | 6 | const plugins = [ 7 | 'transform-decorators-legacy', 8 | ]; 9 | 10 | if (NODE_ENV === 'production') { 11 | plugins.push('transform-react-remove-prop-types'); 12 | } 13 | 14 | if (BUILD_ENV === 'hot') { 15 | plugins.push('react-hot-loader/babel'); 16 | } 17 | 18 | if (BUILD_ENV === 'test') { 19 | plugins.push('dynamic-import-node'); 20 | } 21 | 22 | module.exports = { 23 | presets: [ 24 | ['env', presetOptions], 25 | 'stage-1', 26 | 'react', 27 | ], 28 | plugins, 29 | }; 30 | -------------------------------------------------------------------------------- /webpack.config.js: -------------------------------------------------------------------------------- 1 | const path = require('path'); 2 | const webpack = require('webpack'); 3 | const merge = require('webpack-merge'); 4 | const WebpackNotifierPlugin = require('webpack-notifier'); 5 | const autoprefixer = require('autoprefixer'); 6 | const precss = require('precss'); 7 | const flexbugs = require('postcss-flexbugs-fixes'); 8 | 9 | const libraryName = 'react-signalr'; 10 | 11 | const isProd = process.env.NODE_ENV === 'production'; 12 | 13 | const PATHS = { 14 | root: __dirname, 15 | build: path.join(__dirname, 'lib', 'umd'), 16 | context: path.join(__dirname, 'src'), 17 | jsFileName: isProd ? `${libraryName}.min.js` : `${libraryName}.js`, 18 | entry: path.join(__dirname, 'src', 'index.js'), 19 | }; 20 | 21 | /* 22 | * BASE CONFIG FOR ALL ENVS 23 | */ 24 | const baseConfig = { 25 | context: PATHS.context, 26 | entry: [ 27 | PATHS.entry, 28 | ], 29 | output: { 30 | path: PATHS.build, 31 | filename: PATHS.jsFileName, 32 | library: libraryName, 33 | libraryTarget: 'umd', 34 | umdNamedDefine: true, 35 | }, 36 | module: { 37 | rules: [ 38 | { 39 | test: /(\.jsx|\.js)$/, 40 | exclude: { 41 | test: path.resolve(__dirname, 'node_modules'), 42 | exclude: path.resolve(__dirname, 'node_modules', '@aspnet'), 43 | }, 44 | use: [ 45 | 'babel-loader', 46 | ], 47 | }, 48 | { 49 | test: /\.css$/, 50 | use: [ 51 | 'style-loader', 52 | { 53 | loader: 'css-loader', 54 | options: { 55 | minimize: !!isProd, 56 | }, 57 | }, 58 | { 59 | loader: 'postcss-loader', 60 | options: { 61 | plugins: () => [flexbugs, precss, autoprefixer], 62 | }, 63 | }, 64 | ], 65 | }, 66 | { 67 | test: /\.scss$/, 68 | use: [ 69 | 'style-loader', 70 | { 71 | loader: 'css-loader', 72 | options: { 73 | minimize: !!isProd, 74 | }, 75 | }, 76 | { 77 | loader: 'postcss-loader', 78 | options: { 79 | plugins: () => [flexbugs, precss, autoprefixer], 80 | }, 81 | }, 82 | 'sass-loader', 83 | ], 84 | }, 85 | { 86 | test: /\.svg$/, 87 | exclude: path.resolve(__dirname, 'node_modules', 'font-awesome'), 88 | use: ['babel-loader', 'react-svg-loader'], 89 | }, 90 | { 91 | test: /\.svg$/, 92 | include: path.resolve(__dirname, 'node_modules', 'font-awesome'), 93 | use: [{ 94 | loader: 'file-loader', 95 | options: { 96 | name: '[name].[ext]', 97 | outputPath: 'fonts/', 98 | }, 99 | }], 100 | }, 101 | { 102 | test: /\.(woff|woff2)(\?v=\d+\.\d+\.\d+)?$/, 103 | use: [{ 104 | loader: 'url-loader', 105 | options: { 106 | name: '[name].[ext]', 107 | outputPath: 'fonts/', 108 | limit: 100, 109 | mimetype: 'application/font-woff', 110 | }, 111 | }], 112 | }, 113 | { 114 | test: /\.ttf(\?v=\d+\.\d+\.\d+)?$/, 115 | use: [{ 116 | loader: 'url-loader', 117 | options: { 118 | name: '[name].[ext]', 119 | outputPath: 'fonts/', 120 | limit: 100, 121 | mimetype: 'application/octet-stream', 122 | }, 123 | }], 124 | }, 125 | { 126 | test: /\.eot(\?v=\d+\.\d+\.\d+)?$/, 127 | use: [{ 128 | loader: 'file-loader', 129 | options: { 130 | name: '[name].[ext]', 131 | outputPath: 'fonts/', 132 | }, 133 | }], 134 | }, 135 | { 136 | test: /\.ico$/, 137 | use: [{ 138 | loader: 'file-loader', 139 | options: { 140 | name: '[name].[ext]', 141 | }, 142 | }], 143 | }, 144 | ], 145 | }, 146 | node: { 147 | fs: 'empty', 148 | }, 149 | resolve: { 150 | modules: [ 151 | path.resolve('./src'), 152 | 'node_modules', 153 | ], 154 | extensions: ['.js', '.jsx'], 155 | mainFields: ['es', 'cjs', 'browser', 'module', 'es:next', 'main'], 156 | alias: { 157 | axios: path.resolve('./node_modules/axios'), 158 | react: path.resolve('./node_modules/react'), 159 | 'react-dom': path.resolve('./node_modules/react-dom'), 160 | }, 161 | }, 162 | // Add your peer dependencies here to avoid bundling them to build 163 | externals: { 164 | '@aspnet/signalr-client': '@aspnet/signalr-client', 165 | axios: 'axios', 166 | react: { 167 | root: 'React', 168 | commonjs2: 'react', 169 | commonjs: 'react', 170 | amd: 'react', 171 | umd: 'react', 172 | }, 173 | 'react-dom': { 174 | root: 'ReactDOM', 175 | commonjs2: 'react-dom', 176 | commonjs: 'react-dom', 177 | amd: 'react-dom', 178 | umd: 'react-dom', 179 | }, 180 | }, 181 | }; 182 | 183 | /* 184 | * DEVELOPMENT CONFIG 185 | */ 186 | const devConfig = { 187 | devtool: 'eval-source-map', 188 | plugins: [ 189 | new webpack.DefinePlugin({ 190 | 'process.env': { 191 | NODE_ENV: JSON.stringify('development'), 192 | }, 193 | }), 194 | new WebpackNotifierPlugin(), 195 | new webpack.NamedModulesPlugin(), 196 | ], 197 | }; 198 | 199 | /* 200 | * PRODUCTION CONFIG 201 | */ 202 | const prodConfig = { 203 | devtool: 'source-map', 204 | plugins: [ 205 | new webpack.DefinePlugin({ 206 | 'process.env': { 207 | NODE_ENV: JSON.stringify('production'), 208 | }, 209 | }), 210 | new webpack.optimize.ModuleConcatenationPlugin(), 211 | new webpack.optimize.UglifyJsPlugin({ 212 | sourceMap: true, 213 | output: { 214 | comments: false, 215 | }, 216 | }), 217 | ], 218 | }; 219 | 220 | module.exports = merge(baseConfig, isProd ? prodConfig : devConfig); 221 | --------------------------------------------------------------------------------