@ngxs/websocket-plugin 3.7.3 → 3.7.4
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/bundles/ngxs-websocket-plugin.umd.js +45 -17
- package/bundles/ngxs-websocket-plugin.umd.js.map +1 -1
- package/bundles/ngxs-websocket-plugin.umd.min.js +2 -2
- package/bundles/ngxs-websocket-plugin.umd.min.js.map +1 -1
- package/esm2015/src/websocket-handler.js +36 -14
- package/esm5/src/websocket-handler.js +43 -14
- package/fesm2015/ngxs-websocket-plugin.js +35 -13
- package/fesm2015/ngxs-websocket-plugin.js.map +1 -1
- package/fesm5/ngxs-websocket-plugin.js +42 -13
- package/fesm5/ngxs-websocket-plugin.js.map +1 -1
- package/ngxs-websocket-plugin.metadata.json +1 -1
- package/package.json +4 -4
- package/src/websocket-handler.d.ts +5 -1
- package/tests/typings.d.ts +18 -0
- package/typings.d.ts +0 -0
|
@@ -4,6 +4,7 @@
|
|
|
4
4
|
*/
|
|
5
5
|
import { Injectable, Inject } from '@angular/core';
|
|
6
6
|
import { Actions, Store, getValue, ofActionDispatched } from '@ngxs/store';
|
|
7
|
+
import { Subscription } from 'rxjs';
|
|
7
8
|
import { WebSocketSubject } from 'rxjs/webSocket';
|
|
8
9
|
import { ConnectWebSocket, DisconnectWebSocket, SendWebSocketMessage, NGXS_WEBSOCKET_OPTIONS, WebsocketMessageError, WebSocketDisconnected, TypeKeyPropertyMissingError, WebSocketConnectionUpdated, WebSocketConnected } from './symbols';
|
|
9
10
|
export class WebSocketHandler {
|
|
@@ -44,33 +45,41 @@ export class WebSocketHandler {
|
|
|
44
45
|
}
|
|
45
46
|
};
|
|
46
47
|
this.typeKey = (/** @type {?} */ (this.options.typeKey));
|
|
48
|
+
this.subscription = new Subscription();
|
|
47
49
|
this.setupActionsListeners();
|
|
48
50
|
}
|
|
51
|
+
/**
|
|
52
|
+
* @return {?}
|
|
53
|
+
*/
|
|
54
|
+
ngOnDestroy() {
|
|
55
|
+
this.closeConnection();
|
|
56
|
+
this.subscription.unsubscribe();
|
|
57
|
+
}
|
|
49
58
|
/**
|
|
50
59
|
* @private
|
|
51
60
|
* @return {?}
|
|
52
61
|
*/
|
|
53
62
|
setupActionsListeners() {
|
|
54
|
-
this.actions$.pipe(ofActionDispatched(ConnectWebSocket)).subscribe((/**
|
|
63
|
+
this.subscription.add(this.actions$.pipe(ofActionDispatched(ConnectWebSocket)).subscribe((/**
|
|
55
64
|
* @param {?} __0
|
|
56
65
|
* @return {?}
|
|
57
66
|
*/
|
|
58
67
|
({ payload }) => {
|
|
59
68
|
this.connect(payload);
|
|
60
|
-
}));
|
|
61
|
-
this.actions$.pipe(ofActionDispatched(DisconnectWebSocket)).subscribe((/**
|
|
69
|
+
})));
|
|
70
|
+
this.subscription.add(this.actions$.pipe(ofActionDispatched(DisconnectWebSocket)).subscribe((/**
|
|
62
71
|
* @return {?}
|
|
63
72
|
*/
|
|
64
73
|
() => {
|
|
65
74
|
this.disconnect();
|
|
66
|
-
}));
|
|
67
|
-
this.actions$.pipe(ofActionDispatched(SendWebSocketMessage)).subscribe((/**
|
|
75
|
+
})));
|
|
76
|
+
this.subscription.add(this.actions$.pipe(ofActionDispatched(SendWebSocketMessage)).subscribe((/**
|
|
68
77
|
* @param {?} __0
|
|
69
78
|
* @return {?}
|
|
70
79
|
*/
|
|
71
80
|
({ payload }) => {
|
|
72
81
|
this.send(payload);
|
|
73
|
-
}));
|
|
82
|
+
})));
|
|
74
83
|
}
|
|
75
84
|
/**
|
|
76
85
|
* @private
|
|
@@ -119,11 +128,7 @@ export class WebSocketHandler {
|
|
|
119
128
|
*/
|
|
120
129
|
disconnect() {
|
|
121
130
|
if (this.socket) {
|
|
122
|
-
|
|
123
|
-
// also it doesn't invoke the `onComplete` callback that we passed
|
|
124
|
-
// into `socket.subscribe(...)`
|
|
125
|
-
this.socket.complete();
|
|
126
|
-
this.socket = null;
|
|
131
|
+
this.closeConnection();
|
|
127
132
|
this.dispatchWebSocketDisconnected();
|
|
128
133
|
}
|
|
129
134
|
}
|
|
@@ -165,8 +170,7 @@ export class WebSocketHandler {
|
|
|
165
170
|
*/
|
|
166
171
|
updateConnection() {
|
|
167
172
|
if (this.socket) {
|
|
168
|
-
this.
|
|
169
|
-
this.socket = null;
|
|
173
|
+
this.closeConnection();
|
|
170
174
|
this.store.dispatch(new WebSocketConnectionUpdated());
|
|
171
175
|
}
|
|
172
176
|
}
|
|
@@ -178,6 +182,19 @@ export class WebSocketHandler {
|
|
|
178
182
|
dispatchWebSocketDisconnected() {
|
|
179
183
|
this.store.dispatch(new WebSocketDisconnected());
|
|
180
184
|
}
|
|
185
|
+
/**
|
|
186
|
+
* @private
|
|
187
|
+
* @return {?}
|
|
188
|
+
*/
|
|
189
|
+
closeConnection() {
|
|
190
|
+
// `socket.complete()` closes the connection
|
|
191
|
+
// also it doesn't invoke the `onComplete` callback that we passed
|
|
192
|
+
// into `socket.subscribe(...)`
|
|
193
|
+
if (this.socket !== null) {
|
|
194
|
+
this.socket.complete();
|
|
195
|
+
this.socket = null;
|
|
196
|
+
}
|
|
197
|
+
}
|
|
181
198
|
}
|
|
182
199
|
WebSocketHandler.decorators = [
|
|
183
200
|
{ type: Injectable }
|
|
@@ -204,6 +221,11 @@ if (false) {
|
|
|
204
221
|
* @private
|
|
205
222
|
*/
|
|
206
223
|
WebSocketHandler.prototype.typeKey;
|
|
224
|
+
/**
|
|
225
|
+
* @type {?}
|
|
226
|
+
* @private
|
|
227
|
+
*/
|
|
228
|
+
WebSocketHandler.prototype.subscription;
|
|
207
229
|
/**
|
|
208
230
|
* @type {?}
|
|
209
231
|
* @private
|
|
@@ -220,4 +242,4 @@ if (false) {
|
|
|
220
242
|
*/
|
|
221
243
|
WebSocketHandler.prototype.options;
|
|
222
244
|
}
|
|
223
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"websocket-handler.js","sourceRoot":"ng://@ngxs/websocket-plugin/","sources":["src/websocket-handler.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable, Inject } from '@angular/core';\r\nimport { Actions, Store, getValue, ofActionDispatched } from '@ngxs/store';\r\n\r\nimport { WebSocketSubject, WebSocketSubjectConfig } from 'rxjs/webSocket';\r\n\r\nimport {\r\n  ConnectWebSocket,\r\n  DisconnectWebSocket,\r\n  SendWebSocketMessage,\r\n  NGXS_WEBSOCKET_OPTIONS,\r\n  NgxsWebsocketPluginOptions,\r\n  WebsocketMessageError,\r\n  WebSocketDisconnected,\r\n  TypeKeyPropertyMissingError,\r\n  WebSocketConnectionUpdated,\r\n  WebSocketConnected\r\n} from './symbols';\r\n\r\n@Injectable()\r\nexport class WebSocketHandler {\r\n  private socket: WebSocketSubject<any> | null = null;\r\n\r\n  private config: WebSocketSubjectConfig<any> = {\r\n    url: this.options.url!,\r\n    protocol: this.options.protocol,\r\n    // Default binary type is `blob` for the global `WebSocket`\r\n    binaryType: this.options.binaryType,\r\n    serializer: this.options.serializer,\r\n    deserializer: this.options.deserializer,\r\n    closeObserver: {\r\n      next: () => {\r\n        // ATTENTION!\r\n        // See https://github.com/ReactiveX/rxjs/blob/master/src/internal/observable/dom/WebSocketSubject.ts#L340\r\n        // RxJS socket emits `onComplete` event only if `event.wasClean` is truthy\r\n        // and doesn't complete socket subject if it's falsy\r\n        this.disconnect();\r\n      }\r\n    },\r\n    openObserver: {\r\n      next: () => this.store.dispatch(new WebSocketConnected())\r\n    }\r\n  };\r\n\r\n  private typeKey = this.options.typeKey!;\r\n\r\n  constructor(\r\n    private store: Store,\r\n    private actions$: Actions,\r\n    @Inject(NGXS_WEBSOCKET_OPTIONS) private options: NgxsWebsocketPluginOptions\r\n  ) {\r\n    this.setupActionsListeners();\r\n  }\r\n\r\n  private setupActionsListeners(): void {\r\n    this.actions$.pipe(ofActionDispatched(ConnectWebSocket)).subscribe(({ payload }) => {\r\n      this.connect(payload);\r\n    });\r\n\r\n    this.actions$.pipe(ofActionDispatched(DisconnectWebSocket)).subscribe(() => {\r\n      this.disconnect();\r\n    });\r\n\r\n    this.actions$.pipe(ofActionDispatched(SendWebSocketMessage)).subscribe(({ payload }) => {\r\n      this.send(payload);\r\n    });\r\n  }\r\n\r\n  private connect(options?: NgxsWebsocketPluginOptions): void {\r\n    this.updateConnection();\r\n\r\n    // Users can pass the options in the connect method so\r\n    // if options aren't available at DI bootstrap they have access\r\n    // to pass them here\r\n    if (options) {\r\n      this.mergeConfigWithOptions(options);\r\n    }\r\n\r\n    this.socket = new WebSocketSubject(this.config);\r\n\r\n    this.socket.subscribe({\r\n      next: (message: any) => {\r\n        const type = getValue(message, this.typeKey);\r\n        if (!type) {\r\n          throw new TypeKeyPropertyMissingError(this.typeKey);\r\n        }\r\n        this.store.dispatch({ ...message, type });\r\n      },\r\n      error: (error: any) => {\r\n        if (error instanceof CloseEvent) {\r\n          this.dispatchWebSocketDisconnected();\r\n        } else {\r\n          this.store.dispatch(new WebsocketMessageError(error));\r\n        }\r\n      }\r\n    });\r\n  }\r\n\r\n  private disconnect(): void {\r\n    if (this.socket) {\r\n      // `socket.complete()` closes the connection\r\n      // also it doesn't invoke the `onComplete` callback that we passed\r\n      // into `socket.subscribe(...)`\r\n      this.socket.complete();\r\n      this.socket = null;\r\n      this.dispatchWebSocketDisconnected();\r\n    }\r\n  }\r\n\r\n  private send(data: any): void {\r\n    if (!this.socket) {\r\n      throw new Error('You must connect to the socket before sending any data');\r\n    }\r\n\r\n    this.socket.next(data);\r\n  }\r\n\r\n  /**\r\n   * Don't enlarge the `connect` method\r\n   */\r\n  private mergeConfigWithOptions(options: NgxsWebsocketPluginOptions): void {\r\n    if (options.url) {\r\n      this.config.url = options.url;\r\n    }\r\n\r\n    if (options.serializer) {\r\n      this.config.serializer = options.serializer;\r\n    }\r\n\r\n    if (options.deserializer) {\r\n      this.config.deserializer = options.deserializer;\r\n    }\r\n  }\r\n\r\n  /**\r\n   * To ensure we don't have any memory leaks\r\n   * e.g. if the user occasionally dispatched `ConnectWebSocket` twice\r\n   * then the previous subscription will still live in the memory\r\n   * to prevent such behavior - we close the previous connection if it exists\r\n   */\r\n  private updateConnection(): void {\r\n    if (this.socket) {\r\n      this.socket.complete();\r\n      this.socket = null;\r\n      this.store.dispatch(new WebSocketConnectionUpdated());\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Used in many places so it's better to move the code into function\r\n   */\r\n  private dispatchWebSocketDisconnected(): void {\r\n    this.store.dispatch(new WebSocketDisconnected());\r\n  }\r\n}\r\n"]}
|
|
245
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"websocket-handler.js","sourceRoot":"ng://@ngxs/websocket-plugin/","sources":["src/websocket-handler.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable, Inject, OnDestroy } from '@angular/core';\r\nimport { Actions, Store, getValue, ofActionDispatched } from '@ngxs/store';\r\nimport { Subscription } from 'rxjs';\r\n\r\nimport { WebSocketSubject, WebSocketSubjectConfig } from 'rxjs/webSocket';\r\n\r\nimport {\r\n  ConnectWebSocket,\r\n  DisconnectWebSocket,\r\n  SendWebSocketMessage,\r\n  NGXS_WEBSOCKET_OPTIONS,\r\n  NgxsWebsocketPluginOptions,\r\n  WebsocketMessageError,\r\n  WebSocketDisconnected,\r\n  TypeKeyPropertyMissingError,\r\n  WebSocketConnectionUpdated,\r\n  WebSocketConnected\r\n} from './symbols';\r\n\r\n@Injectable()\r\nexport class WebSocketHandler implements OnDestroy {\r\n  private socket: WebSocketSubject<any> | null = null;\r\n\r\n  private config: WebSocketSubjectConfig<any> = {\r\n    url: this.options.url!,\r\n    protocol: this.options.protocol,\r\n    // Default binary type is `blob` for the global `WebSocket`\r\n    binaryType: this.options.binaryType,\r\n    serializer: this.options.serializer,\r\n    deserializer: this.options.deserializer,\r\n    closeObserver: {\r\n      next: () => {\r\n        // ATTENTION!\r\n        // See https://github.com/ReactiveX/rxjs/blob/master/src/internal/observable/dom/WebSocketSubject.ts#L340\r\n        // RxJS socket emits `onComplete` event only if `event.wasClean` is truthy\r\n        // and doesn't complete socket subject if it's falsy\r\n        this.disconnect();\r\n      }\r\n    },\r\n    openObserver: {\r\n      next: () => this.store.dispatch(new WebSocketConnected())\r\n    }\r\n  };\r\n\r\n  private typeKey = this.options.typeKey!;\r\n\r\n  private subscription = new Subscription();\r\n\r\n  constructor(\r\n    private store: Store,\r\n    private actions$: Actions,\r\n    @Inject(NGXS_WEBSOCKET_OPTIONS) private options: NgxsWebsocketPluginOptions\r\n  ) {\r\n    this.setupActionsListeners();\r\n  }\r\n\r\n  ngOnDestroy(): void {\r\n    this.closeConnection();\r\n    this.subscription.unsubscribe();\r\n  }\r\n\r\n  private setupActionsListeners(): void {\r\n    this.subscription.add(\r\n      this.actions$.pipe(ofActionDispatched(ConnectWebSocket)).subscribe(({ payload }) => {\r\n        this.connect(payload);\r\n      })\r\n    );\r\n\r\n    this.subscription.add(\r\n      this.actions$.pipe(ofActionDispatched(DisconnectWebSocket)).subscribe(() => {\r\n        this.disconnect();\r\n      })\r\n    );\r\n\r\n    this.subscription.add(\r\n      this.actions$.pipe(ofActionDispatched(SendWebSocketMessage)).subscribe(({ payload }) => {\r\n        this.send(payload);\r\n      })\r\n    );\r\n  }\r\n\r\n  private connect(options?: NgxsWebsocketPluginOptions): void {\r\n    this.updateConnection();\r\n\r\n    // Users can pass the options in the connect method so\r\n    // if options aren't available at DI bootstrap they have access\r\n    // to pass them here\r\n    if (options) {\r\n      this.mergeConfigWithOptions(options);\r\n    }\r\n\r\n    this.socket = new WebSocketSubject(this.config);\r\n\r\n    this.socket.subscribe({\r\n      next: (message: any) => {\r\n        const type = getValue(message, this.typeKey);\r\n        if (!type) {\r\n          throw new TypeKeyPropertyMissingError(this.typeKey);\r\n        }\r\n        this.store.dispatch({ ...message, type });\r\n      },\r\n      error: (error: any) => {\r\n        if (error instanceof CloseEvent) {\r\n          this.dispatchWebSocketDisconnected();\r\n        } else {\r\n          this.store.dispatch(new WebsocketMessageError(error));\r\n        }\r\n      }\r\n    });\r\n  }\r\n\r\n  private disconnect(): void {\r\n    if (this.socket) {\r\n      this.closeConnection();\r\n      this.dispatchWebSocketDisconnected();\r\n    }\r\n  }\r\n\r\n  private send(data: any): void {\r\n    if (!this.socket) {\r\n      throw new Error('You must connect to the socket before sending any data');\r\n    }\r\n\r\n    this.socket.next(data);\r\n  }\r\n\r\n  /**\r\n   * Don't enlarge the `connect` method\r\n   */\r\n  private mergeConfigWithOptions(options: NgxsWebsocketPluginOptions): void {\r\n    if (options.url) {\r\n      this.config.url = options.url;\r\n    }\r\n\r\n    if (options.serializer) {\r\n      this.config.serializer = options.serializer;\r\n    }\r\n\r\n    if (options.deserializer) {\r\n      this.config.deserializer = options.deserializer;\r\n    }\r\n  }\r\n\r\n  /**\r\n   * To ensure we don't have any memory leaks\r\n   * e.g. if the user occasionally dispatched `ConnectWebSocket` twice\r\n   * then the previous subscription will still live in the memory\r\n   * to prevent such behavior - we close the previous connection if it exists\r\n   */\r\n  private updateConnection(): void {\r\n    if (this.socket) {\r\n      this.closeConnection();\r\n      this.store.dispatch(new WebSocketConnectionUpdated());\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Used in many places so it's better to move the code into function\r\n   */\r\n  private dispatchWebSocketDisconnected(): void {\r\n    this.store.dispatch(new WebSocketDisconnected());\r\n  }\r\n\r\n  private closeConnection(): void {\r\n    // `socket.complete()` closes the connection\r\n    // also it doesn't invoke the `onComplete` callback that we passed\r\n    // into `socket.subscribe(...)`\r\n    if (this.socket !== null) {\r\n      this.socket.complete();\r\n      this.socket = null;\r\n    }\r\n  }\r\n}\r\n"]}
|
|
@@ -5,6 +5,7 @@
|
|
|
5
5
|
import * as tslib_1 from "tslib";
|
|
6
6
|
import { Injectable, Inject } from '@angular/core';
|
|
7
7
|
import { Actions, Store, getValue, ofActionDispatched } from '@ngxs/store';
|
|
8
|
+
import { Subscription } from 'rxjs';
|
|
8
9
|
import { WebSocketSubject } from 'rxjs/webSocket';
|
|
9
10
|
import { ConnectWebSocket, DisconnectWebSocket, SendWebSocketMessage, NGXS_WEBSOCKET_OPTIONS, WebsocketMessageError, WebSocketDisconnected, TypeKeyPropertyMissingError, WebSocketConnectionUpdated, WebSocketConnected } from './symbols';
|
|
10
11
|
var WebSocketHandler = /** @class */ (function () {
|
|
@@ -41,8 +42,19 @@ var WebSocketHandler = /** @class */ (function () {
|
|
|
41
42
|
}
|
|
42
43
|
};
|
|
43
44
|
this.typeKey = (/** @type {?} */ (this.options.typeKey));
|
|
45
|
+
this.subscription = new Subscription();
|
|
44
46
|
this.setupActionsListeners();
|
|
45
47
|
}
|
|
48
|
+
/**
|
|
49
|
+
* @return {?}
|
|
50
|
+
*/
|
|
51
|
+
WebSocketHandler.prototype.ngOnDestroy = /**
|
|
52
|
+
* @return {?}
|
|
53
|
+
*/
|
|
54
|
+
function () {
|
|
55
|
+
this.closeConnection();
|
|
56
|
+
this.subscription.unsubscribe();
|
|
57
|
+
};
|
|
46
58
|
/**
|
|
47
59
|
* @private
|
|
48
60
|
* @return {?}
|
|
@@ -53,28 +65,28 @@ var WebSocketHandler = /** @class */ (function () {
|
|
|
53
65
|
*/
|
|
54
66
|
function () {
|
|
55
67
|
var _this = this;
|
|
56
|
-
this.actions$.pipe(ofActionDispatched(ConnectWebSocket)).subscribe((/**
|
|
68
|
+
this.subscription.add(this.actions$.pipe(ofActionDispatched(ConnectWebSocket)).subscribe((/**
|
|
57
69
|
* @param {?} __0
|
|
58
70
|
* @return {?}
|
|
59
71
|
*/
|
|
60
72
|
function (_a) {
|
|
61
73
|
var payload = _a.payload;
|
|
62
74
|
_this.connect(payload);
|
|
63
|
-
}));
|
|
64
|
-
this.actions$.pipe(ofActionDispatched(DisconnectWebSocket)).subscribe((/**
|
|
75
|
+
})));
|
|
76
|
+
this.subscription.add(this.actions$.pipe(ofActionDispatched(DisconnectWebSocket)).subscribe((/**
|
|
65
77
|
* @return {?}
|
|
66
78
|
*/
|
|
67
79
|
function () {
|
|
68
80
|
_this.disconnect();
|
|
69
|
-
}));
|
|
70
|
-
this.actions$.pipe(ofActionDispatched(SendWebSocketMessage)).subscribe((/**
|
|
81
|
+
})));
|
|
82
|
+
this.subscription.add(this.actions$.pipe(ofActionDispatched(SendWebSocketMessage)).subscribe((/**
|
|
71
83
|
* @param {?} __0
|
|
72
84
|
* @return {?}
|
|
73
85
|
*/
|
|
74
86
|
function (_a) {
|
|
75
87
|
var payload = _a.payload;
|
|
76
88
|
_this.send(payload);
|
|
77
|
-
}));
|
|
89
|
+
})));
|
|
78
90
|
};
|
|
79
91
|
/**
|
|
80
92
|
* @private
|
|
@@ -133,11 +145,7 @@ var WebSocketHandler = /** @class */ (function () {
|
|
|
133
145
|
*/
|
|
134
146
|
function () {
|
|
135
147
|
if (this.socket) {
|
|
136
|
-
|
|
137
|
-
// also it doesn't invoke the `onComplete` callback that we passed
|
|
138
|
-
// into `socket.subscribe(...)`
|
|
139
|
-
this.socket.complete();
|
|
140
|
-
this.socket = null;
|
|
148
|
+
this.closeConnection();
|
|
141
149
|
this.dispatchWebSocketDisconnected();
|
|
142
150
|
}
|
|
143
151
|
};
|
|
@@ -207,8 +215,7 @@ var WebSocketHandler = /** @class */ (function () {
|
|
|
207
215
|
*/
|
|
208
216
|
function () {
|
|
209
217
|
if (this.socket) {
|
|
210
|
-
this.
|
|
211
|
-
this.socket = null;
|
|
218
|
+
this.closeConnection();
|
|
212
219
|
this.store.dispatch(new WebSocketConnectionUpdated());
|
|
213
220
|
}
|
|
214
221
|
};
|
|
@@ -228,6 +235,23 @@ var WebSocketHandler = /** @class */ (function () {
|
|
|
228
235
|
function () {
|
|
229
236
|
this.store.dispatch(new WebSocketDisconnected());
|
|
230
237
|
};
|
|
238
|
+
/**
|
|
239
|
+
* @private
|
|
240
|
+
* @return {?}
|
|
241
|
+
*/
|
|
242
|
+
WebSocketHandler.prototype.closeConnection = /**
|
|
243
|
+
* @private
|
|
244
|
+
* @return {?}
|
|
245
|
+
*/
|
|
246
|
+
function () {
|
|
247
|
+
// `socket.complete()` closes the connection
|
|
248
|
+
// also it doesn't invoke the `onComplete` callback that we passed
|
|
249
|
+
// into `socket.subscribe(...)`
|
|
250
|
+
if (this.socket !== null) {
|
|
251
|
+
this.socket.complete();
|
|
252
|
+
this.socket = null;
|
|
253
|
+
}
|
|
254
|
+
};
|
|
231
255
|
WebSocketHandler.decorators = [
|
|
232
256
|
{ type: Injectable }
|
|
233
257
|
];
|
|
@@ -256,6 +280,11 @@ if (false) {
|
|
|
256
280
|
* @private
|
|
257
281
|
*/
|
|
258
282
|
WebSocketHandler.prototype.typeKey;
|
|
283
|
+
/**
|
|
284
|
+
* @type {?}
|
|
285
|
+
* @private
|
|
286
|
+
*/
|
|
287
|
+
WebSocketHandler.prototype.subscription;
|
|
259
288
|
/**
|
|
260
289
|
* @type {?}
|
|
261
290
|
* @private
|
|
@@ -272,4 +301,4 @@ if (false) {
|
|
|
272
301
|
*/
|
|
273
302
|
WebSocketHandler.prototype.options;
|
|
274
303
|
}
|
|
275
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"websocket-handler.js","sourceRoot":"ng://@ngxs/websocket-plugin/","sources":["src/websocket-handler.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable, Inject } from '@angular/core';\r\nimport { Actions, Store, getValue, ofActionDispatched } from '@ngxs/store';\r\n\r\nimport { WebSocketSubject, WebSocketSubjectConfig } from 'rxjs/webSocket';\r\n\r\nimport {\r\n  ConnectWebSocket,\r\n  DisconnectWebSocket,\r\n  SendWebSocketMessage,\r\n  NGXS_WEBSOCKET_OPTIONS,\r\n  NgxsWebsocketPluginOptions,\r\n  WebsocketMessageError,\r\n  WebSocketDisconnected,\r\n  TypeKeyPropertyMissingError,\r\n  WebSocketConnectionUpdated,\r\n  WebSocketConnected\r\n} from './symbols';\r\n\r\n@Injectable()\r\nexport class WebSocketHandler {\r\n  private socket: WebSocketSubject<any> | null = null;\r\n\r\n  private config: WebSocketSubjectConfig<any> = {\r\n    url: this.options.url!,\r\n    protocol: this.options.protocol,\r\n    // Default binary type is `blob` for the global `WebSocket`\r\n    binaryType: this.options.binaryType,\r\n    serializer: this.options.serializer,\r\n    deserializer: this.options.deserializer,\r\n    closeObserver: {\r\n      next: () => {\r\n        // ATTENTION!\r\n        // See https://github.com/ReactiveX/rxjs/blob/master/src/internal/observable/dom/WebSocketSubject.ts#L340\r\n        // RxJS socket emits `onComplete` event only if `event.wasClean` is truthy\r\n        // and doesn't complete socket subject if it's falsy\r\n        this.disconnect();\r\n      }\r\n    },\r\n    openObserver: {\r\n      next: () => this.store.dispatch(new WebSocketConnected())\r\n    }\r\n  };\r\n\r\n  private typeKey = this.options.typeKey!;\r\n\r\n  constructor(\r\n    private store: Store,\r\n    private actions$: Actions,\r\n    @Inject(NGXS_WEBSOCKET_OPTIONS) private options: NgxsWebsocketPluginOptions\r\n  ) {\r\n    this.setupActionsListeners();\r\n  }\r\n\r\n  private setupActionsListeners(): void {\r\n    this.actions$.pipe(ofActionDispatched(ConnectWebSocket)).subscribe(({ payload }) => {\r\n      this.connect(payload);\r\n    });\r\n\r\n    this.actions$.pipe(ofActionDispatched(DisconnectWebSocket)).subscribe(() => {\r\n      this.disconnect();\r\n    });\r\n\r\n    this.actions$.pipe(ofActionDispatched(SendWebSocketMessage)).subscribe(({ payload }) => {\r\n      this.send(payload);\r\n    });\r\n  }\r\n\r\n  private connect(options?: NgxsWebsocketPluginOptions): void {\r\n    this.updateConnection();\r\n\r\n    // Users can pass the options in the connect method so\r\n    // if options aren't available at DI bootstrap they have access\r\n    // to pass them here\r\n    if (options) {\r\n      this.mergeConfigWithOptions(options);\r\n    }\r\n\r\n    this.socket = new WebSocketSubject(this.config);\r\n\r\n    this.socket.subscribe({\r\n      next: (message: any) => {\r\n        const type = getValue(message, this.typeKey);\r\n        if (!type) {\r\n          throw new TypeKeyPropertyMissingError(this.typeKey);\r\n        }\r\n        this.store.dispatch({ ...message, type });\r\n      },\r\n      error: (error: any) => {\r\n        if (error instanceof CloseEvent) {\r\n          this.dispatchWebSocketDisconnected();\r\n        } else {\r\n          this.store.dispatch(new WebsocketMessageError(error));\r\n        }\r\n      }\r\n    });\r\n  }\r\n\r\n  private disconnect(): void {\r\n    if (this.socket) {\r\n      // `socket.complete()` closes the connection\r\n      // also it doesn't invoke the `onComplete` callback that we passed\r\n      // into `socket.subscribe(...)`\r\n      this.socket.complete();\r\n      this.socket = null;\r\n      this.dispatchWebSocketDisconnected();\r\n    }\r\n  }\r\n\r\n  private send(data: any): void {\r\n    if (!this.socket) {\r\n      throw new Error('You must connect to the socket before sending any data');\r\n    }\r\n\r\n    this.socket.next(data);\r\n  }\r\n\r\n  /**\r\n   * Don't enlarge the `connect` method\r\n   */\r\n  private mergeConfigWithOptions(options: NgxsWebsocketPluginOptions): void {\r\n    if (options.url) {\r\n      this.config.url = options.url;\r\n    }\r\n\r\n    if (options.serializer) {\r\n      this.config.serializer = options.serializer;\r\n    }\r\n\r\n    if (options.deserializer) {\r\n      this.config.deserializer = options.deserializer;\r\n    }\r\n  }\r\n\r\n  /**\r\n   * To ensure we don't have any memory leaks\r\n   * e.g. if the user occasionally dispatched `ConnectWebSocket` twice\r\n   * then the previous subscription will still live in the memory\r\n   * to prevent such behavior - we close the previous connection if it exists\r\n   */\r\n  private updateConnection(): void {\r\n    if (this.socket) {\r\n      this.socket.complete();\r\n      this.socket = null;\r\n      this.store.dispatch(new WebSocketConnectionUpdated());\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Used in many places so it's better to move the code into function\r\n   */\r\n  private dispatchWebSocketDisconnected(): void {\r\n    this.store.dispatch(new WebSocketDisconnected());\r\n  }\r\n}\r\n"]}
|
|
304
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"websocket-handler.js","sourceRoot":"ng://@ngxs/websocket-plugin/","sources":["src/websocket-handler.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable, Inject, OnDestroy } from '@angular/core';\r\nimport { Actions, Store, getValue, ofActionDispatched } from '@ngxs/store';\r\nimport { Subscription } from 'rxjs';\r\n\r\nimport { WebSocketSubject, WebSocketSubjectConfig } from 'rxjs/webSocket';\r\n\r\nimport {\r\n  ConnectWebSocket,\r\n  DisconnectWebSocket,\r\n  SendWebSocketMessage,\r\n  NGXS_WEBSOCKET_OPTIONS,\r\n  NgxsWebsocketPluginOptions,\r\n  WebsocketMessageError,\r\n  WebSocketDisconnected,\r\n  TypeKeyPropertyMissingError,\r\n  WebSocketConnectionUpdated,\r\n  WebSocketConnected\r\n} from './symbols';\r\n\r\n@Injectable()\r\nexport class WebSocketHandler implements OnDestroy {\r\n  private socket: WebSocketSubject<any> | null = null;\r\n\r\n  private config: WebSocketSubjectConfig<any> = {\r\n    url: this.options.url!,\r\n    protocol: this.options.protocol,\r\n    // Default binary type is `blob` for the global `WebSocket`\r\n    binaryType: this.options.binaryType,\r\n    serializer: this.options.serializer,\r\n    deserializer: this.options.deserializer,\r\n    closeObserver: {\r\n      next: () => {\r\n        // ATTENTION!\r\n        // See https://github.com/ReactiveX/rxjs/blob/master/src/internal/observable/dom/WebSocketSubject.ts#L340\r\n        // RxJS socket emits `onComplete` event only if `event.wasClean` is truthy\r\n        // and doesn't complete socket subject if it's falsy\r\n        this.disconnect();\r\n      }\r\n    },\r\n    openObserver: {\r\n      next: () => this.store.dispatch(new WebSocketConnected())\r\n    }\r\n  };\r\n\r\n  private typeKey = this.options.typeKey!;\r\n\r\n  private subscription = new Subscription();\r\n\r\n  constructor(\r\n    private store: Store,\r\n    private actions$: Actions,\r\n    @Inject(NGXS_WEBSOCKET_OPTIONS) private options: NgxsWebsocketPluginOptions\r\n  ) {\r\n    this.setupActionsListeners();\r\n  }\r\n\r\n  ngOnDestroy(): void {\r\n    this.closeConnection();\r\n    this.subscription.unsubscribe();\r\n  }\r\n\r\n  private setupActionsListeners(): void {\r\n    this.subscription.add(\r\n      this.actions$.pipe(ofActionDispatched(ConnectWebSocket)).subscribe(({ payload }) => {\r\n        this.connect(payload);\r\n      })\r\n    );\r\n\r\n    this.subscription.add(\r\n      this.actions$.pipe(ofActionDispatched(DisconnectWebSocket)).subscribe(() => {\r\n        this.disconnect();\r\n      })\r\n    );\r\n\r\n    this.subscription.add(\r\n      this.actions$.pipe(ofActionDispatched(SendWebSocketMessage)).subscribe(({ payload }) => {\r\n        this.send(payload);\r\n      })\r\n    );\r\n  }\r\n\r\n  private connect(options?: NgxsWebsocketPluginOptions): void {\r\n    this.updateConnection();\r\n\r\n    // Users can pass the options in the connect method so\r\n    // if options aren't available at DI bootstrap they have access\r\n    // to pass them here\r\n    if (options) {\r\n      this.mergeConfigWithOptions(options);\r\n    }\r\n\r\n    this.socket = new WebSocketSubject(this.config);\r\n\r\n    this.socket.subscribe({\r\n      next: (message: any) => {\r\n        const type = getValue(message, this.typeKey);\r\n        if (!type) {\r\n          throw new TypeKeyPropertyMissingError(this.typeKey);\r\n        }\r\n        this.store.dispatch({ ...message, type });\r\n      },\r\n      error: (error: any) => {\r\n        if (error instanceof CloseEvent) {\r\n          this.dispatchWebSocketDisconnected();\r\n        } else {\r\n          this.store.dispatch(new WebsocketMessageError(error));\r\n        }\r\n      }\r\n    });\r\n  }\r\n\r\n  private disconnect(): void {\r\n    if (this.socket) {\r\n      this.closeConnection();\r\n      this.dispatchWebSocketDisconnected();\r\n    }\r\n  }\r\n\r\n  private send(data: any): void {\r\n    if (!this.socket) {\r\n      throw new Error('You must connect to the socket before sending any data');\r\n    }\r\n\r\n    this.socket.next(data);\r\n  }\r\n\r\n  /**\r\n   * Don't enlarge the `connect` method\r\n   */\r\n  private mergeConfigWithOptions(options: NgxsWebsocketPluginOptions): void {\r\n    if (options.url) {\r\n      this.config.url = options.url;\r\n    }\r\n\r\n    if (options.serializer) {\r\n      this.config.serializer = options.serializer;\r\n    }\r\n\r\n    if (options.deserializer) {\r\n      this.config.deserializer = options.deserializer;\r\n    }\r\n  }\r\n\r\n  /**\r\n   * To ensure we don't have any memory leaks\r\n   * e.g. if the user occasionally dispatched `ConnectWebSocket` twice\r\n   * then the previous subscription will still live in the memory\r\n   * to prevent such behavior - we close the previous connection if it exists\r\n   */\r\n  private updateConnection(): void {\r\n    if (this.socket) {\r\n      this.closeConnection();\r\n      this.store.dispatch(new WebSocketConnectionUpdated());\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Used in many places so it's better to move the code into function\r\n   */\r\n  private dispatchWebSocketDisconnected(): void {\r\n    this.store.dispatch(new WebSocketDisconnected());\r\n  }\r\n\r\n  private closeConnection(): void {\r\n    // `socket.complete()` closes the connection\r\n    // also it doesn't invoke the `onComplete` callback that we passed\r\n    // into `socket.subscribe(...)`\r\n    if (this.socket !== null) {\r\n      this.socket.complete();\r\n      this.socket = null;\r\n    }\r\n  }\r\n}\r\n"]}
|
|
@@ -1,5 +1,6 @@
|
|
|
1
1
|
import { InjectionToken, Injectable, Inject, APP_INITIALIZER, NgModule } from '@angular/core';
|
|
2
2
|
import { ofActionDispatched, getValue, Store, Actions } from '@ngxs/store';
|
|
3
|
+
import { Subscription } from 'rxjs';
|
|
3
4
|
import { WebSocketSubject } from 'rxjs/webSocket';
|
|
4
5
|
|
|
5
6
|
/**
|
|
@@ -240,33 +241,41 @@ class WebSocketHandler {
|
|
|
240
241
|
}
|
|
241
242
|
};
|
|
242
243
|
this.typeKey = (/** @type {?} */ (this.options.typeKey));
|
|
244
|
+
this.subscription = new Subscription();
|
|
243
245
|
this.setupActionsListeners();
|
|
244
246
|
}
|
|
247
|
+
/**
|
|
248
|
+
* @return {?}
|
|
249
|
+
*/
|
|
250
|
+
ngOnDestroy() {
|
|
251
|
+
this.closeConnection();
|
|
252
|
+
this.subscription.unsubscribe();
|
|
253
|
+
}
|
|
245
254
|
/**
|
|
246
255
|
* @private
|
|
247
256
|
* @return {?}
|
|
248
257
|
*/
|
|
249
258
|
setupActionsListeners() {
|
|
250
|
-
this.actions$.pipe(ofActionDispatched(ConnectWebSocket)).subscribe((/**
|
|
259
|
+
this.subscription.add(this.actions$.pipe(ofActionDispatched(ConnectWebSocket)).subscribe((/**
|
|
251
260
|
* @param {?} __0
|
|
252
261
|
* @return {?}
|
|
253
262
|
*/
|
|
254
263
|
({ payload }) => {
|
|
255
264
|
this.connect(payload);
|
|
256
|
-
}));
|
|
257
|
-
this.actions$.pipe(ofActionDispatched(DisconnectWebSocket)).subscribe((/**
|
|
265
|
+
})));
|
|
266
|
+
this.subscription.add(this.actions$.pipe(ofActionDispatched(DisconnectWebSocket)).subscribe((/**
|
|
258
267
|
* @return {?}
|
|
259
268
|
*/
|
|
260
269
|
() => {
|
|
261
270
|
this.disconnect();
|
|
262
|
-
}));
|
|
263
|
-
this.actions$.pipe(ofActionDispatched(SendWebSocketMessage)).subscribe((/**
|
|
271
|
+
})));
|
|
272
|
+
this.subscription.add(this.actions$.pipe(ofActionDispatched(SendWebSocketMessage)).subscribe((/**
|
|
264
273
|
* @param {?} __0
|
|
265
274
|
* @return {?}
|
|
266
275
|
*/
|
|
267
276
|
({ payload }) => {
|
|
268
277
|
this.send(payload);
|
|
269
|
-
}));
|
|
278
|
+
})));
|
|
270
279
|
}
|
|
271
280
|
/**
|
|
272
281
|
* @private
|
|
@@ -315,11 +324,7 @@ class WebSocketHandler {
|
|
|
315
324
|
*/
|
|
316
325
|
disconnect() {
|
|
317
326
|
if (this.socket) {
|
|
318
|
-
|
|
319
|
-
// also it doesn't invoke the `onComplete` callback that we passed
|
|
320
|
-
// into `socket.subscribe(...)`
|
|
321
|
-
this.socket.complete();
|
|
322
|
-
this.socket = null;
|
|
327
|
+
this.closeConnection();
|
|
323
328
|
this.dispatchWebSocketDisconnected();
|
|
324
329
|
}
|
|
325
330
|
}
|
|
@@ -361,8 +366,7 @@ class WebSocketHandler {
|
|
|
361
366
|
*/
|
|
362
367
|
updateConnection() {
|
|
363
368
|
if (this.socket) {
|
|
364
|
-
this.
|
|
365
|
-
this.socket = null;
|
|
369
|
+
this.closeConnection();
|
|
366
370
|
this.store.dispatch(new WebSocketConnectionUpdated());
|
|
367
371
|
}
|
|
368
372
|
}
|
|
@@ -374,6 +378,19 @@ class WebSocketHandler {
|
|
|
374
378
|
dispatchWebSocketDisconnected() {
|
|
375
379
|
this.store.dispatch(new WebSocketDisconnected());
|
|
376
380
|
}
|
|
381
|
+
/**
|
|
382
|
+
* @private
|
|
383
|
+
* @return {?}
|
|
384
|
+
*/
|
|
385
|
+
closeConnection() {
|
|
386
|
+
// `socket.complete()` closes the connection
|
|
387
|
+
// also it doesn't invoke the `onComplete` callback that we passed
|
|
388
|
+
// into `socket.subscribe(...)`
|
|
389
|
+
if (this.socket !== null) {
|
|
390
|
+
this.socket.complete();
|
|
391
|
+
this.socket = null;
|
|
392
|
+
}
|
|
393
|
+
}
|
|
377
394
|
}
|
|
378
395
|
WebSocketHandler.decorators = [
|
|
379
396
|
{ type: Injectable }
|
|
@@ -400,6 +417,11 @@ if (false) {
|
|
|
400
417
|
* @private
|
|
401
418
|
*/
|
|
402
419
|
WebSocketHandler.prototype.typeKey;
|
|
420
|
+
/**
|
|
421
|
+
* @type {?}
|
|
422
|
+
* @private
|
|
423
|
+
*/
|
|
424
|
+
WebSocketHandler.prototype.subscription;
|
|
403
425
|
/**
|
|
404
426
|
* @type {?}
|
|
405
427
|
* @private
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"ngxs-websocket-plugin.js","sources":["ng://@ngxs/websocket-plugin/src/symbols.ts","ng://@ngxs/websocket-plugin/src/websocket-handler.ts","ng://@ngxs/websocket-plugin/src/websocket.module.ts"],"sourcesContent":["import { InjectionToken } from '@angular/core';\r\n\r\nexport const NGXS_WEBSOCKET_OPTIONS = new InjectionToken('NGXS_WEBSOCKET_OPTIONS');\r\n\r\nexport interface NgxsWebsocketPluginOptions {\r\n /**\r\n * URL of the websocket.\r\n */\r\n url?: string;\r\n\r\n /**\r\n * Either a single protocol string or an array of protocol strings.\r\n * These strings are used to indicate sub-protocols, so that a single server\r\n * can implement multiple WebSocket sub-protocols (for example, you might want one server to be able\r\n * to handle different types of interactions depending on the specified protocol).\r\n * If you don't specify a protocol string, an empty string is assumed.\r\n */\r\n protocol?: string | string[];\r\n\r\n /**\r\n * Sets the `binaryType` property of the underlying WebSocket.\r\n */\r\n binaryType?: 'blob' | 'arraybuffer';\r\n\r\n /**\r\n * The property name to distigunish this type for the store.\r\n * Default: 'type'\r\n */\r\n typeKey?: string;\r\n\r\n /**\r\n * Interval to try and reconnect.\r\n * Default: 5000\r\n */\r\n reconnectInterval?: number;\r\n\r\n /**\r\n * Number of reconnect attemps.\r\n * Default: 10\r\n */\r\n reconnectAttempts?: number;\r\n\r\n /**\r\n * Serializer to call before sending messages\r\n * Default: `json.stringify`\r\n */\r\n serializer?: (data: any) => string;\r\n\r\n /**\r\n * Deseralizer before publishing the message.\r\n */\r\n deserializer?: (e: MessageEvent) => any;\r\n}\r\n\r\nexport function noop(..._args: any[]) {\r\n return function() {};\r\n}\r\n\r\n/**\r\n * Action to connect to the websocket. Optionally pass a URL.\r\n */\r\nexport class ConnectWebSocket {\r\n static get type() {\r\n // NOTE: Not necessary to declare the type in this way in your code. See https://github.com/ngxs/store/pull/644#issuecomment-436003138\r\n return '[WebSocket] Connect';\r\n }\r\n constructor(public payload?: NgxsWebsocketPluginOptions) {}\r\n}\r\n\r\n/**\r\n * Action triggered when a error ocurrs\r\n */\r\nexport class WebsocketMessageError {\r\n static get type() {\r\n // NOTE: Not necessary to declare the type in this way in your code. See https://github.com/ngxs/store/pull/644#issuecomment-436003138\r\n return '[WebSocket] Message Error';\r\n }\r\n constructor(public payload: any) {}\r\n}\r\n\r\n/**\r\n * Action to disconnect the websocket.\r\n */\r\nexport class DisconnectWebSocket {\r\n static get type() {\r\n // NOTE: Not necessary to declare the type in this way in your code. See https://github.com/ngxs/store/pull/644#issuecomment-436003138\r\n return '[WebSocket] Disconnect';\r\n }\r\n}\r\n\r\n/**\r\n * Action triggered when websocket is connected\r\n */\r\nexport class WebSocketConnected {\r\n static get type() {\r\n return '[WebSocket] Connected';\r\n }\r\n}\r\n\r\n/**\r\n * Action triggered when websocket is disconnected\r\n */\r\nexport class WebSocketDisconnected {\r\n static get type() {\r\n // NOTE: Not necessary to declare the type in this way in your code. See https://github.com/ngxs/store/pull/644#issuecomment-436003138\r\n return '[WebSocket] Disconnected';\r\n }\r\n}\r\n\r\n/**\r\n * Action to send to the server.\r\n */\r\nexport class SendWebSocketMessage {\r\n static get type() {\r\n // NOTE: Not necessary to declare the type in this way in your code. See https://github.com/ngxs/store/pull/644#issuecomment-436003138\r\n return '[WebSocket] Send Message';\r\n }\r\n constructor(public payload: any) {}\r\n}\r\n\r\n/**\r\n * Action dispatched when the user tries to connect if the connection already exists.\r\n */\r\nexport class WebSocketConnectionUpdated {\r\n static get type() {\r\n // NOTE: Not necessary to declare the type in this way in your code. See https://github.com/ngxs/store/pull/644#issuecomment-436003138\r\n return '[WebSocket] Connection Updated';\r\n }\r\n}\r\n\r\n/**\r\n * This error is thrown where there is no `type` (or custom `typeKey`) property\r\n * on the message that came from the server side socket\r\n */\r\nexport class TypeKeyPropertyMissingError extends Error {\r\n constructor(typeKey: string) {\r\n super(`Property ${typeKey} is missing on the socket message`);\r\n }\r\n}\r\n","import { Injectable, Inject } from '@angular/core';\r\nimport { Actions, Store, getValue, ofActionDispatched } from '@ngxs/store';\r\n\r\nimport { WebSocketSubject, WebSocketSubjectConfig } from 'rxjs/webSocket';\r\n\r\nimport {\r\n ConnectWebSocket,\r\n DisconnectWebSocket,\r\n SendWebSocketMessage,\r\n NGXS_WEBSOCKET_OPTIONS,\r\n NgxsWebsocketPluginOptions,\r\n WebsocketMessageError,\r\n WebSocketDisconnected,\r\n TypeKeyPropertyMissingError,\r\n WebSocketConnectionUpdated,\r\n WebSocketConnected\r\n} from './symbols';\r\n\r\n@Injectable()\r\nexport class WebSocketHandler {\r\n private socket: WebSocketSubject<any> | null = null;\r\n\r\n private config: WebSocketSubjectConfig<any> = {\r\n url: this.options.url!,\r\n protocol: this.options.protocol,\r\n // Default binary type is `blob` for the global `WebSocket`\r\n binaryType: this.options.binaryType,\r\n serializer: this.options.serializer,\r\n deserializer: this.options.deserializer,\r\n closeObserver: {\r\n next: () => {\r\n // ATTENTION!\r\n // See https://github.com/ReactiveX/rxjs/blob/master/src/internal/observable/dom/WebSocketSubject.ts#L340\r\n // RxJS socket emits `onComplete` event only if `event.wasClean` is truthy\r\n // and doesn't complete socket subject if it's falsy\r\n this.disconnect();\r\n }\r\n },\r\n openObserver: {\r\n next: () => this.store.dispatch(new WebSocketConnected())\r\n }\r\n };\r\n\r\n private typeKey = this.options.typeKey!;\r\n\r\n constructor(\r\n private store: Store,\r\n private actions$: Actions,\r\n @Inject(NGXS_WEBSOCKET_OPTIONS) private options: NgxsWebsocketPluginOptions\r\n ) {\r\n this.setupActionsListeners();\r\n }\r\n\r\n private setupActionsListeners(): void {\r\n this.actions$.pipe(ofActionDispatched(ConnectWebSocket)).subscribe(({ payload }) => {\r\n this.connect(payload);\r\n });\r\n\r\n this.actions$.pipe(ofActionDispatched(DisconnectWebSocket)).subscribe(() => {\r\n this.disconnect();\r\n });\r\n\r\n this.actions$.pipe(ofActionDispatched(SendWebSocketMessage)).subscribe(({ payload }) => {\r\n this.send(payload);\r\n });\r\n }\r\n\r\n private connect(options?: NgxsWebsocketPluginOptions): void {\r\n this.updateConnection();\r\n\r\n // Users can pass the options in the connect method so\r\n // if options aren't available at DI bootstrap they have access\r\n // to pass them here\r\n if (options) {\r\n this.mergeConfigWithOptions(options);\r\n }\r\n\r\n this.socket = new WebSocketSubject(this.config);\r\n\r\n this.socket.subscribe({\r\n next: (message: any) => {\r\n const type = getValue(message, this.typeKey);\r\n if (!type) {\r\n throw new TypeKeyPropertyMissingError(this.typeKey);\r\n }\r\n this.store.dispatch({ ...message, type });\r\n },\r\n error: (error: any) => {\r\n if (error instanceof CloseEvent) {\r\n this.dispatchWebSocketDisconnected();\r\n } else {\r\n this.store.dispatch(new WebsocketMessageError(error));\r\n }\r\n }\r\n });\r\n }\r\n\r\n private disconnect(): void {\r\n if (this.socket) {\r\n // `socket.complete()` closes the connection\r\n // also it doesn't invoke the `onComplete` callback that we passed\r\n // into `socket.subscribe(...)`\r\n this.socket.complete();\r\n this.socket = null;\r\n this.dispatchWebSocketDisconnected();\r\n }\r\n }\r\n\r\n private send(data: any): void {\r\n if (!this.socket) {\r\n throw new Error('You must connect to the socket before sending any data');\r\n }\r\n\r\n this.socket.next(data);\r\n }\r\n\r\n /**\r\n * Don't enlarge the `connect` method\r\n */\r\n private mergeConfigWithOptions(options: NgxsWebsocketPluginOptions): void {\r\n if (options.url) {\r\n this.config.url = options.url;\r\n }\r\n\r\n if (options.serializer) {\r\n this.config.serializer = options.serializer;\r\n }\r\n\r\n if (options.deserializer) {\r\n this.config.deserializer = options.deserializer;\r\n }\r\n }\r\n\r\n /**\r\n * To ensure we don't have any memory leaks\r\n * e.g. if the user occasionally dispatched `ConnectWebSocket` twice\r\n * then the previous subscription will still live in the memory\r\n * to prevent such behavior - we close the previous connection if it exists\r\n */\r\n private updateConnection(): void {\r\n if (this.socket) {\r\n this.socket.complete();\r\n this.socket = null;\r\n this.store.dispatch(new WebSocketConnectionUpdated());\r\n }\r\n }\r\n\r\n /**\r\n * Used in many places so it's better to move the code into function\r\n */\r\n private dispatchWebSocketDisconnected(): void {\r\n this.store.dispatch(new WebSocketDisconnected());\r\n }\r\n}\r\n","import { NgModule, ModuleWithProviders, APP_INITIALIZER, InjectionToken } from '@angular/core';\r\n\r\nimport { WebSocketHandler } from './websocket-handler';\r\nimport { NgxsWebsocketPluginOptions, NGXS_WEBSOCKET_OPTIONS, noop } from './symbols';\r\n\r\nexport function websocketOptionsFactory(options: NgxsWebsocketPluginOptions) {\r\n return {\r\n reconnectInterval: 5000,\r\n reconnectAttempts: 10,\r\n typeKey: 'type',\r\n deserializer(e: MessageEvent) {\r\n return JSON.parse(e.data);\r\n },\r\n serializer(value: any) {\r\n return JSON.stringify(value);\r\n },\r\n ...options\r\n };\r\n}\r\n\r\nexport const USER_OPTIONS = new InjectionToken('USER_OPTIONS');\r\n\r\n@NgModule()\r\nexport class NgxsWebsocketPluginModule {\r\n static forRoot(\r\n options?: NgxsWebsocketPluginOptions\r\n ): ModuleWithProviders<NgxsWebsocketPluginModule> {\r\n return {\r\n ngModule: NgxsWebsocketPluginModule,\r\n providers: [\r\n WebSocketHandler,\r\n {\r\n provide: USER_OPTIONS,\r\n useValue: options\r\n },\r\n {\r\n provide: NGXS_WEBSOCKET_OPTIONS,\r\n useFactory: websocketOptionsFactory,\r\n deps: [USER_OPTIONS]\r\n },\r\n {\r\n provide: APP_INITIALIZER,\r\n useFactory: noop,\r\n deps: [WebSocketHandler],\r\n multi: true\r\n }\r\n ]\r\n };\r\n }\r\n}\r\n"],"names":[],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"ngxs-websocket-plugin.js","sources":["ng://@ngxs/websocket-plugin/src/symbols.ts","ng://@ngxs/websocket-plugin/src/websocket-handler.ts","ng://@ngxs/websocket-plugin/src/websocket.module.ts"],"sourcesContent":["import { InjectionToken } from '@angular/core';\r\n\r\nexport const NGXS_WEBSOCKET_OPTIONS = new InjectionToken('NGXS_WEBSOCKET_OPTIONS');\r\n\r\nexport interface NgxsWebsocketPluginOptions {\r\n /**\r\n * URL of the websocket.\r\n */\r\n url?: string;\r\n\r\n /**\r\n * Either a single protocol string or an array of protocol strings.\r\n * These strings are used to indicate sub-protocols, so that a single server\r\n * can implement multiple WebSocket sub-protocols (for example, you might want one server to be able\r\n * to handle different types of interactions depending on the specified protocol).\r\n * If you don't specify a protocol string, an empty string is assumed.\r\n */\r\n protocol?: string | string[];\r\n\r\n /**\r\n * Sets the `binaryType` property of the underlying WebSocket.\r\n */\r\n binaryType?: 'blob' | 'arraybuffer';\r\n\r\n /**\r\n * The property name to distigunish this type for the store.\r\n * Default: 'type'\r\n */\r\n typeKey?: string;\r\n\r\n /**\r\n * Interval to try and reconnect.\r\n * Default: 5000\r\n */\r\n reconnectInterval?: number;\r\n\r\n /**\r\n * Number of reconnect attemps.\r\n * Default: 10\r\n */\r\n reconnectAttempts?: number;\r\n\r\n /**\r\n * Serializer to call before sending messages\r\n * Default: `json.stringify`\r\n */\r\n serializer?: (data: any) => string;\r\n\r\n /**\r\n * Deseralizer before publishing the message.\r\n */\r\n deserializer?: (e: MessageEvent) => any;\r\n}\r\n\r\nexport function noop(..._args: any[]) {\r\n return function() {};\r\n}\r\n\r\n/**\r\n * Action to connect to the websocket. Optionally pass a URL.\r\n */\r\nexport class ConnectWebSocket {\r\n static get type() {\r\n // NOTE: Not necessary to declare the type in this way in your code. See https://github.com/ngxs/store/pull/644#issuecomment-436003138\r\n return '[WebSocket] Connect';\r\n }\r\n constructor(public payload?: NgxsWebsocketPluginOptions) {}\r\n}\r\n\r\n/**\r\n * Action triggered when a error ocurrs\r\n */\r\nexport class WebsocketMessageError {\r\n static get type() {\r\n // NOTE: Not necessary to declare the type in this way in your code. See https://github.com/ngxs/store/pull/644#issuecomment-436003138\r\n return '[WebSocket] Message Error';\r\n }\r\n constructor(public payload: any) {}\r\n}\r\n\r\n/**\r\n * Action to disconnect the websocket.\r\n */\r\nexport class DisconnectWebSocket {\r\n static get type() {\r\n // NOTE: Not necessary to declare the type in this way in your code. See https://github.com/ngxs/store/pull/644#issuecomment-436003138\r\n return '[WebSocket] Disconnect';\r\n }\r\n}\r\n\r\n/**\r\n * Action triggered when websocket is connected\r\n */\r\nexport class WebSocketConnected {\r\n static get type() {\r\n return '[WebSocket] Connected';\r\n }\r\n}\r\n\r\n/**\r\n * Action triggered when websocket is disconnected\r\n */\r\nexport class WebSocketDisconnected {\r\n static get type() {\r\n // NOTE: Not necessary to declare the type in this way in your code. See https://github.com/ngxs/store/pull/644#issuecomment-436003138\r\n return '[WebSocket] Disconnected';\r\n }\r\n}\r\n\r\n/**\r\n * Action to send to the server.\r\n */\r\nexport class SendWebSocketMessage {\r\n static get type() {\r\n // NOTE: Not necessary to declare the type in this way in your code. See https://github.com/ngxs/store/pull/644#issuecomment-436003138\r\n return '[WebSocket] Send Message';\r\n }\r\n constructor(public payload: any) {}\r\n}\r\n\r\n/**\r\n * Action dispatched when the user tries to connect if the connection already exists.\r\n */\r\nexport class WebSocketConnectionUpdated {\r\n static get type() {\r\n // NOTE: Not necessary to declare the type in this way in your code. See https://github.com/ngxs/store/pull/644#issuecomment-436003138\r\n return '[WebSocket] Connection Updated';\r\n }\r\n}\r\n\r\n/**\r\n * This error is thrown where there is no `type` (or custom `typeKey`) property\r\n * on the message that came from the server side socket\r\n */\r\nexport class TypeKeyPropertyMissingError extends Error {\r\n constructor(typeKey: string) {\r\n super(`Property ${typeKey} is missing on the socket message`);\r\n }\r\n}\r\n","import { Injectable, Inject, OnDestroy } from '@angular/core';\r\nimport { Actions, Store, getValue, ofActionDispatched } from '@ngxs/store';\r\nimport { Subscription } from 'rxjs';\r\n\r\nimport { WebSocketSubject, WebSocketSubjectConfig } from 'rxjs/webSocket';\r\n\r\nimport {\r\n ConnectWebSocket,\r\n DisconnectWebSocket,\r\n SendWebSocketMessage,\r\n NGXS_WEBSOCKET_OPTIONS,\r\n NgxsWebsocketPluginOptions,\r\n WebsocketMessageError,\r\n WebSocketDisconnected,\r\n TypeKeyPropertyMissingError,\r\n WebSocketConnectionUpdated,\r\n WebSocketConnected\r\n} from './symbols';\r\n\r\n@Injectable()\r\nexport class WebSocketHandler implements OnDestroy {\r\n private socket: WebSocketSubject<any> | null = null;\r\n\r\n private config: WebSocketSubjectConfig<any> = {\r\n url: this.options.url!,\r\n protocol: this.options.protocol,\r\n // Default binary type is `blob` for the global `WebSocket`\r\n binaryType: this.options.binaryType,\r\n serializer: this.options.serializer,\r\n deserializer: this.options.deserializer,\r\n closeObserver: {\r\n next: () => {\r\n // ATTENTION!\r\n // See https://github.com/ReactiveX/rxjs/blob/master/src/internal/observable/dom/WebSocketSubject.ts#L340\r\n // RxJS socket emits `onComplete` event only if `event.wasClean` is truthy\r\n // and doesn't complete socket subject if it's falsy\r\n this.disconnect();\r\n }\r\n },\r\n openObserver: {\r\n next: () => this.store.dispatch(new WebSocketConnected())\r\n }\r\n };\r\n\r\n private typeKey = this.options.typeKey!;\r\n\r\n private subscription = new Subscription();\r\n\r\n constructor(\r\n private store: Store,\r\n private actions$: Actions,\r\n @Inject(NGXS_WEBSOCKET_OPTIONS) private options: NgxsWebsocketPluginOptions\r\n ) {\r\n this.setupActionsListeners();\r\n }\r\n\r\n ngOnDestroy(): void {\r\n this.closeConnection();\r\n this.subscription.unsubscribe();\r\n }\r\n\r\n private setupActionsListeners(): void {\r\n this.subscription.add(\r\n this.actions$.pipe(ofActionDispatched(ConnectWebSocket)).subscribe(({ payload }) => {\r\n this.connect(payload);\r\n })\r\n );\r\n\r\n this.subscription.add(\r\n this.actions$.pipe(ofActionDispatched(DisconnectWebSocket)).subscribe(() => {\r\n this.disconnect();\r\n })\r\n );\r\n\r\n this.subscription.add(\r\n this.actions$.pipe(ofActionDispatched(SendWebSocketMessage)).subscribe(({ payload }) => {\r\n this.send(payload);\r\n })\r\n );\r\n }\r\n\r\n private connect(options?: NgxsWebsocketPluginOptions): void {\r\n this.updateConnection();\r\n\r\n // Users can pass the options in the connect method so\r\n // if options aren't available at DI bootstrap they have access\r\n // to pass them here\r\n if (options) {\r\n this.mergeConfigWithOptions(options);\r\n }\r\n\r\n this.socket = new WebSocketSubject(this.config);\r\n\r\n this.socket.subscribe({\r\n next: (message: any) => {\r\n const type = getValue(message, this.typeKey);\r\n if (!type) {\r\n throw new TypeKeyPropertyMissingError(this.typeKey);\r\n }\r\n this.store.dispatch({ ...message, type });\r\n },\r\n error: (error: any) => {\r\n if (error instanceof CloseEvent) {\r\n this.dispatchWebSocketDisconnected();\r\n } else {\r\n this.store.dispatch(new WebsocketMessageError(error));\r\n }\r\n }\r\n });\r\n }\r\n\r\n private disconnect(): void {\r\n if (this.socket) {\r\n this.closeConnection();\r\n this.dispatchWebSocketDisconnected();\r\n }\r\n }\r\n\r\n private send(data: any): void {\r\n if (!this.socket) {\r\n throw new Error('You must connect to the socket before sending any data');\r\n }\r\n\r\n this.socket.next(data);\r\n }\r\n\r\n /**\r\n * Don't enlarge the `connect` method\r\n */\r\n private mergeConfigWithOptions(options: NgxsWebsocketPluginOptions): void {\r\n if (options.url) {\r\n this.config.url = options.url;\r\n }\r\n\r\n if (options.serializer) {\r\n this.config.serializer = options.serializer;\r\n }\r\n\r\n if (options.deserializer) {\r\n this.config.deserializer = options.deserializer;\r\n }\r\n }\r\n\r\n /**\r\n * To ensure we don't have any memory leaks\r\n * e.g. if the user occasionally dispatched `ConnectWebSocket` twice\r\n * then the previous subscription will still live in the memory\r\n * to prevent such behavior - we close the previous connection if it exists\r\n */\r\n private updateConnection(): void {\r\n if (this.socket) {\r\n this.closeConnection();\r\n this.store.dispatch(new WebSocketConnectionUpdated());\r\n }\r\n }\r\n\r\n /**\r\n * Used in many places so it's better to move the code into function\r\n */\r\n private dispatchWebSocketDisconnected(): void {\r\n this.store.dispatch(new WebSocketDisconnected());\r\n }\r\n\r\n private closeConnection(): void {\r\n // `socket.complete()` closes the connection\r\n // also it doesn't invoke the `onComplete` callback that we passed\r\n // into `socket.subscribe(...)`\r\n if (this.socket !== null) {\r\n this.socket.complete();\r\n this.socket = null;\r\n }\r\n }\r\n}\r\n","import { NgModule, ModuleWithProviders, APP_INITIALIZER, InjectionToken } from '@angular/core';\r\n\r\nimport { WebSocketHandler } from './websocket-handler';\r\nimport { NgxsWebsocketPluginOptions, NGXS_WEBSOCKET_OPTIONS, noop } from './symbols';\r\n\r\nexport function websocketOptionsFactory(options: NgxsWebsocketPluginOptions) {\r\n return {\r\n reconnectInterval: 5000,\r\n reconnectAttempts: 10,\r\n typeKey: 'type',\r\n deserializer(e: MessageEvent) {\r\n return JSON.parse(e.data);\r\n },\r\n serializer(value: any) {\r\n return JSON.stringify(value);\r\n },\r\n ...options\r\n };\r\n}\r\n\r\nexport const USER_OPTIONS = new InjectionToken('USER_OPTIONS');\r\n\r\n@NgModule()\r\nexport class NgxsWebsocketPluginModule {\r\n static forRoot(\r\n options?: NgxsWebsocketPluginOptions\r\n ): ModuleWithProviders<NgxsWebsocketPluginModule> {\r\n return {\r\n ngModule: NgxsWebsocketPluginModule,\r\n providers: [\r\n WebSocketHandler,\r\n {\r\n provide: USER_OPTIONS,\r\n useValue: options\r\n },\r\n {\r\n provide: NGXS_WEBSOCKET_OPTIONS,\r\n useFactory: websocketOptionsFactory,\r\n deps: [USER_OPTIONS]\r\n },\r\n {\r\n provide: APP_INITIALIZER,\r\n useFactory: noop,\r\n deps: [WebSocketHandler],\r\n multi: true\r\n }\r\n ]\r\n };\r\n }\r\n}\r\n"],"names":[],"mappings":"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"}
|
|
@@ -1,6 +1,7 @@
|
|
|
1
1
|
import { __extends, __assign } from 'tslib';
|
|
2
2
|
import { InjectionToken, Injectable, Inject, APP_INITIALIZER, NgModule } from '@angular/core';
|
|
3
3
|
import { ofActionDispatched, getValue, Store, Actions } from '@ngxs/store';
|
|
4
|
+
import { Subscription } from 'rxjs';
|
|
4
5
|
import { WebSocketSubject } from 'rxjs/webSocket';
|
|
5
6
|
|
|
6
7
|
/**
|
|
@@ -299,8 +300,19 @@ var WebSocketHandler = /** @class */ (function () {
|
|
|
299
300
|
}
|
|
300
301
|
};
|
|
301
302
|
this.typeKey = (/** @type {?} */ (this.options.typeKey));
|
|
303
|
+
this.subscription = new Subscription();
|
|
302
304
|
this.setupActionsListeners();
|
|
303
305
|
}
|
|
306
|
+
/**
|
|
307
|
+
* @return {?}
|
|
308
|
+
*/
|
|
309
|
+
WebSocketHandler.prototype.ngOnDestroy = /**
|
|
310
|
+
* @return {?}
|
|
311
|
+
*/
|
|
312
|
+
function () {
|
|
313
|
+
this.closeConnection();
|
|
314
|
+
this.subscription.unsubscribe();
|
|
315
|
+
};
|
|
304
316
|
/**
|
|
305
317
|
* @private
|
|
306
318
|
* @return {?}
|
|
@@ -311,28 +323,28 @@ var WebSocketHandler = /** @class */ (function () {
|
|
|
311
323
|
*/
|
|
312
324
|
function () {
|
|
313
325
|
var _this = this;
|
|
314
|
-
this.actions$.pipe(ofActionDispatched(ConnectWebSocket)).subscribe((/**
|
|
326
|
+
this.subscription.add(this.actions$.pipe(ofActionDispatched(ConnectWebSocket)).subscribe((/**
|
|
315
327
|
* @param {?} __0
|
|
316
328
|
* @return {?}
|
|
317
329
|
*/
|
|
318
330
|
function (_a) {
|
|
319
331
|
var payload = _a.payload;
|
|
320
332
|
_this.connect(payload);
|
|
321
|
-
}));
|
|
322
|
-
this.actions$.pipe(ofActionDispatched(DisconnectWebSocket)).subscribe((/**
|
|
333
|
+
})));
|
|
334
|
+
this.subscription.add(this.actions$.pipe(ofActionDispatched(DisconnectWebSocket)).subscribe((/**
|
|
323
335
|
* @return {?}
|
|
324
336
|
*/
|
|
325
337
|
function () {
|
|
326
338
|
_this.disconnect();
|
|
327
|
-
}));
|
|
328
|
-
this.actions$.pipe(ofActionDispatched(SendWebSocketMessage)).subscribe((/**
|
|
339
|
+
})));
|
|
340
|
+
this.subscription.add(this.actions$.pipe(ofActionDispatched(SendWebSocketMessage)).subscribe((/**
|
|
329
341
|
* @param {?} __0
|
|
330
342
|
* @return {?}
|
|
331
343
|
*/
|
|
332
344
|
function (_a) {
|
|
333
345
|
var payload = _a.payload;
|
|
334
346
|
_this.send(payload);
|
|
335
|
-
}));
|
|
347
|
+
})));
|
|
336
348
|
};
|
|
337
349
|
/**
|
|
338
350
|
* @private
|
|
@@ -391,11 +403,7 @@ var WebSocketHandler = /** @class */ (function () {
|
|
|
391
403
|
*/
|
|
392
404
|
function () {
|
|
393
405
|
if (this.socket) {
|
|
394
|
-
|
|
395
|
-
// also it doesn't invoke the `onComplete` callback that we passed
|
|
396
|
-
// into `socket.subscribe(...)`
|
|
397
|
-
this.socket.complete();
|
|
398
|
-
this.socket = null;
|
|
406
|
+
this.closeConnection();
|
|
399
407
|
this.dispatchWebSocketDisconnected();
|
|
400
408
|
}
|
|
401
409
|
};
|
|
@@ -465,8 +473,7 @@ var WebSocketHandler = /** @class */ (function () {
|
|
|
465
473
|
*/
|
|
466
474
|
function () {
|
|
467
475
|
if (this.socket) {
|
|
468
|
-
this.
|
|
469
|
-
this.socket = null;
|
|
476
|
+
this.closeConnection();
|
|
470
477
|
this.store.dispatch(new WebSocketConnectionUpdated());
|
|
471
478
|
}
|
|
472
479
|
};
|
|
@@ -486,6 +493,23 @@ var WebSocketHandler = /** @class */ (function () {
|
|
|
486
493
|
function () {
|
|
487
494
|
this.store.dispatch(new WebSocketDisconnected());
|
|
488
495
|
};
|
|
496
|
+
/**
|
|
497
|
+
* @private
|
|
498
|
+
* @return {?}
|
|
499
|
+
*/
|
|
500
|
+
WebSocketHandler.prototype.closeConnection = /**
|
|
501
|
+
* @private
|
|
502
|
+
* @return {?}
|
|
503
|
+
*/
|
|
504
|
+
function () {
|
|
505
|
+
// `socket.complete()` closes the connection
|
|
506
|
+
// also it doesn't invoke the `onComplete` callback that we passed
|
|
507
|
+
// into `socket.subscribe(...)`
|
|
508
|
+
if (this.socket !== null) {
|
|
509
|
+
this.socket.complete();
|
|
510
|
+
this.socket = null;
|
|
511
|
+
}
|
|
512
|
+
};
|
|
489
513
|
WebSocketHandler.decorators = [
|
|
490
514
|
{ type: Injectable }
|
|
491
515
|
];
|
|
@@ -513,6 +537,11 @@ if (false) {
|
|
|
513
537
|
* @private
|
|
514
538
|
*/
|
|
515
539
|
WebSocketHandler.prototype.typeKey;
|
|
540
|
+
/**
|
|
541
|
+
* @type {?}
|
|
542
|
+
* @private
|
|
543
|
+
*/
|
|
544
|
+
WebSocketHandler.prototype.subscription;
|
|
516
545
|
/**
|
|
517
546
|
* @type {?}
|
|
518
547
|
* @private
|