@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.
@@ -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
- // `socket.complete()` closes the connection
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.socket.complete();
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":";;;;AAAA,OAAO,EAAE,UAAU,EAAE,MAAM,EAAa,MAAM,eAAe,CAAC;AAC9D,OAAO,EAAE,OAAO,EAAE,KAAK,EAAE,QAAQ,EAAE,kBAAkB,EAAE,MAAM,aAAa,CAAC;AAC3E,OAAO,EAAE,YAAY,EAAE,MAAM,MAAM,CAAC;AAEpC,OAAO,EAAE,gBAAgB,EAA0B,MAAM,gBAAgB,CAAC;AAE1E,OAAO,EACL,gBAAgB,EAChB,mBAAmB,EACnB,oBAAoB,EACpB,sBAAsB,EAEtB,qBAAqB,EACrB,qBAAqB,EACrB,2BAA2B,EAC3B,0BAA0B,EAC1B,kBAAkB,EACnB,MAAM,WAAW,CAAC;AAGnB,MAAM,OAAO,gBAAgB;;;;;;IA4B3B,YACU,KAAY,EACZ,QAAiB,EACe,OAAmC;QAFnE,UAAK,GAAL,KAAK,CAAO;QACZ,aAAQ,GAAR,QAAQ,CAAS;QACe,YAAO,GAAP,OAAO,CAA4B;QA9BrE,WAAM,GAAiC,IAAI,CAAC;QAE5C,WAAM,GAAgC;YAC5C,GAAG,EAAE,mBAAA,IAAI,CAAC,OAAO,CAAC,GAAG,EAAC;YACtB,QAAQ,EAAE,IAAI,CAAC,OAAO,CAAC,QAAQ;;YAE/B,UAAU,EAAE,IAAI,CAAC,OAAO,CAAC,UAAU;YACnC,UAAU,EAAE,IAAI,CAAC,OAAO,CAAC,UAAU;YACnC,YAAY,EAAE,IAAI,CAAC,OAAO,CAAC,YAAY;YACvC,aAAa,EAAE;gBACb,IAAI;;;gBAAE,GAAG,EAAE;oBACT,aAAa;oBACb,yGAAyG;oBACzG,0EAA0E;oBAC1E,oDAAoD;oBACpD,IAAI,CAAC,UAAU,EAAE,CAAC;gBACpB,CAAC,CAAA;aACF;YACD,YAAY,EAAE;gBACZ,IAAI;;;gBAAE,GAAG,EAAE,CAAC,IAAI,CAAC,KAAK,CAAC,QAAQ,CAAC,IAAI,kBAAkB,EAAE,CAAC,CAAA;aAC1D;SACF,CAAC;QAEM,YAAO,GAAG,mBAAA,IAAI,CAAC,OAAO,CAAC,OAAO,EAAC,CAAC;QAEhC,iBAAY,GAAG,IAAI,YAAY,EAAE,CAAC;QAOxC,IAAI,CAAC,qBAAqB,EAAE,CAAC;IAC/B,CAAC;;;;IAED,WAAW;QACT,IAAI,CAAC,eAAe,EAAE,CAAC;QACvB,IAAI,CAAC,YAAY,CAAC,WAAW,EAAE,CAAC;IAClC,CAAC;;;;;IAEO,qBAAqB;QAC3B,IAAI,CAAC,YAAY,CAAC,GAAG,CACnB,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,kBAAkB,CAAC,gBAAgB,CAAC,CAAC,CAAC,SAAS;;;;QAAC,CAAC,EAAE,OAAO,EAAE,EAAE,EAAE;YACjF,IAAI,CAAC,OAAO,CAAC,OAAO,CAAC,CAAC;QACxB,CAAC,EAAC,CACH,CAAC;QAEF,IAAI,CAAC,YAAY,CAAC,GAAG,CACnB,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,kBAAkB,CAAC,mBAAmB,CAAC,CAAC,CAAC,SAAS;;;QAAC,GAAG,EAAE;YACzE,IAAI,CAAC,UAAU,EAAE,CAAC;QACpB,CAAC,EAAC,CACH,CAAC;QAEF,IAAI,CAAC,YAAY,CAAC,GAAG,CACnB,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,kBAAkB,CAAC,oBAAoB,CAAC,CAAC,CAAC,SAAS;;;;QAAC,CAAC,EAAE,OAAO,EAAE,EAAE,EAAE;YACrF,IAAI,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;QACrB,CAAC,EAAC,CACH,CAAC;IACJ,CAAC;;;;;;IAEO,OAAO,CAAC,OAAoC;QAClD,IAAI,CAAC,gBAAgB,EAAE,CAAC;QAExB,sDAAsD;QACtD,+DAA+D;QAC/D,oBAAoB;QACpB,IAAI,OAAO,EAAE;YACX,IAAI,CAAC,sBAAsB,CAAC,OAAO,CAAC,CAAC;SACtC;QAED,IAAI,CAAC,MAAM,GAAG,IAAI,gBAAgB,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;QAEhD,IAAI,CAAC,MAAM,CAAC,SAAS,CAAC;YACpB,IAAI;;;;YAAE,CAAC,OAAY,EAAE,EAAE;;sBACf,IAAI,GAAG,QAAQ,CAAC,OAAO,EAAE,IAAI,CAAC,OAAO,CAAC;gBAC5C,IAAI,CAAC,IAAI,EAAE;oBACT,MAAM,IAAI,2BAA2B,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;iBACrD;gBACD,IAAI,CAAC,KAAK,CAAC,QAAQ,mBAAM,OAAO,IAAE,IAAI,IAAG,CAAC;YAC5C,CAAC,CAAA;YACD,KAAK;;;;YAAE,CAAC,KAAU,EAAE,EAAE;gBACpB,IAAI,KAAK,YAAY,UAAU,EAAE;oBAC/B,IAAI,CAAC,6BAA6B,EAAE,CAAC;iBACtC;qBAAM;oBACL,IAAI,CAAC,KAAK,CAAC,QAAQ,CAAC,IAAI,qBAAqB,CAAC,KAAK,CAAC,CAAC,CAAC;iBACvD;YACH,CAAC,CAAA;SACF,CAAC,CAAC;IACL,CAAC;;;;;IAEO,UAAU;QAChB,IAAI,IAAI,CAAC,MAAM,EAAE;YACf,IAAI,CAAC,eAAe,EAAE,CAAC;YACvB,IAAI,CAAC,6BAA6B,EAAE,CAAC;SACtC;IACH,CAAC;;;;;;IAEO,IAAI,CAAC,IAAS;QACpB,IAAI,CAAC,IAAI,CAAC,MAAM,EAAE;YAChB,MAAM,IAAI,KAAK,CAAC,wDAAwD,CAAC,CAAC;SAC3E;QAED,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;IACzB,CAAC;;;;;;;IAKO,sBAAsB,CAAC,OAAmC;QAChE,IAAI,OAAO,CAAC,GAAG,EAAE;YACf,IAAI,CAAC,MAAM,CAAC,GAAG,GAAG,OAAO,CAAC,GAAG,CAAC;SAC/B;QAED,IAAI,OAAO,CAAC,UAAU,EAAE;YACtB,IAAI,CAAC,MAAM,CAAC,UAAU,GAAG,OAAO,CAAC,UAAU,CAAC;SAC7C;QAED,IAAI,OAAO,CAAC,YAAY,EAAE;YACxB,IAAI,CAAC,MAAM,CAAC,YAAY,GAAG,OAAO,CAAC,YAAY,CAAC;SACjD;IACH,CAAC;;;;;;;;;IAQO,gBAAgB;QACtB,IAAI,IAAI,CAAC,MAAM,EAAE;YACf,IAAI,CAAC,eAAe,EAAE,CAAC;YACvB,IAAI,CAAC,KAAK,CAAC,QAAQ,CAAC,IAAI,0BAA0B,EAAE,CAAC,CAAC;SACvD;IACH,CAAC;;;;;;IAKO,6BAA6B;QACnC,IAAI,CAAC,KAAK,CAAC,QAAQ,CAAC,IAAI,qBAAqB,EAAE,CAAC,CAAC;IACnD,CAAC;;;;;IAEO,eAAe;QACrB,4CAA4C;QAC5C,kEAAkE;QAClE,+BAA+B;QAC/B,IAAI,IAAI,CAAC,MAAM,KAAK,IAAI,EAAE;YACxB,IAAI,CAAC,MAAM,CAAC,QAAQ,EAAE,CAAC;YACvB,IAAI,CAAC,MAAM,GAAG,IAAI,CAAC;SACpB;IACH,CAAC;;;YAxJF,UAAU;;;;YAlBO,KAAK;YAAd,OAAO;4CAkDX,MAAM,SAAC,sBAAsB;;;;;;;IA9BhC,kCAAoD;;;;;IAEpD,kCAmBE;;;;;IAEF,mCAAwC;;;;;IAExC,wCAA0C;;;;;IAGxC,iCAAoB;;;;;IACpB,oCAAyB;;;;;IACzB,mCAA2E","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
- // `socket.complete()` closes the connection
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.socket.complete();
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
- // `socket.complete()` closes the connection
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.socket.complete();
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
- // `socket.complete()` closes the connection
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.socket.complete();
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