@elderbyte/ngx-starter 17.7.1 → 17.7.3

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.
@@ -1,7 +1,7 @@
1
1
  import { EventStreamContentType, fetchEventSource } from '@microsoft/fetch-event-source';
2
2
  import { LoggerFactory } from '@elderbyte/ts-logger';
3
3
  import { BehaviorSubject, Subject } from 'rxjs';
4
- import { map } from 'rxjs/operators';
4
+ import { filter } from 'rxjs/operators';
5
5
  export var ReactiveEventSourceState;
6
6
  (function (ReactiveEventSourceState) {
7
7
  ReactiveEventSourceState["IDLE"] = "IDLE";
@@ -11,12 +11,18 @@ export var ReactiveEventSourceState;
11
11
  ReactiveEventSourceState["ERROR"] = "ERROR";
12
12
  })(ReactiveEventSourceState || (ReactiveEventSourceState = {}));
13
13
  export class ReactiveSSeMessage {
14
- constructor(id, event, data, retry) {
14
+ constructor(id, event, dataRaw, retry) {
15
15
  this.id = id;
16
16
  this.event = event;
17
- this.data = data;
17
+ this.dataRaw = dataRaw;
18
18
  this.retry = retry;
19
19
  }
20
+ json() {
21
+ return JSON.parse(this.dataRaw);
22
+ }
23
+ jsonAs() {
24
+ return this.json();
25
+ }
20
26
  }
21
27
  export class ReactiveFetchEventSource {
22
28
  /***************************************************************************
@@ -45,10 +51,12 @@ export class ReactiveFetchEventSource {
45
51
  * Fields *
46
52
  * *
47
53
  **************************************************************************/
54
+ this.EVENT_TYPE_DEFAULT = 'message';
55
+ this.RECONNECT_DELAY_MS = 3000;
48
56
  this.log = LoggerFactory.getLogger(this.constructor.name);
49
- this.abortController = new AbortController();
57
+ this.abortController = null;
50
58
  this.state$ = new BehaviorSubject(ReactiveEventSourceState.IDLE);
51
- this.eventTopics = new Map();
59
+ this.eventRelay = new Subject();
52
60
  if (!requestInitProvider) {
53
61
  throw new Error('You must provide a event source url provider!');
54
62
  }
@@ -82,22 +90,15 @@ export class ReactiveFetchEventSource {
82
90
  * Get an event stream for the given event type.
83
91
  * @param eventTypeRaw The event type. Defaults to 'message'.
84
92
  */
85
- events(eventTypeRaw) {
93
+ streamEventsOfType(eventTypeRaw) {
86
94
  const eventType = this.eventTypeOrDefault(eventTypeRaw);
87
- if (!this.eventTopics.has(eventType)) {
88
- this.eventTopics.set(eventType, new Subject());
89
- }
90
- const eventSubj = this.eventTopics.get(eventType);
91
- return eventSubj.asObservable();
95
+ return this.streamEvents().pipe(filter(e => e.event == eventType));
92
96
  }
93
97
  /**
94
- * Get an event stream of messages parsed from json.
95
- * (event.data must be in json format)
96
- *
97
- * @param eventType The event type. Defaults to 'message'.
98
+ * Get an event stream for all events no matter their type.
98
99
  */
99
- eventsJson(eventType) {
100
- return this.events(eventType).pipe(map(event => JSON.parse(event.data)));
100
+ streamEvents() {
101
+ return this.eventRelay.asObservable();
101
102
  }
102
103
  /**
103
104
  * Close this event source. It won't reconnect.
@@ -133,7 +134,10 @@ export class ReactiveFetchEventSource {
133
134
  */
134
135
  closeCurrent(reason) {
135
136
  this.log.debug('Closing the event-source, reason: ' + reason);
136
- this.abortController.abort(reason);
137
+ const crt = this.abortController;
138
+ if (crt) {
139
+ crt.abort(reason);
140
+ }
137
141
  }
138
142
  /**
139
143
  * Reconnects this event source, unless closed is true;
@@ -148,9 +152,10 @@ export class ReactiveFetchEventSource {
148
152
  }
149
153
  openEventSource() {
150
154
  const requestInit = this.requestInitProvider();
155
+ const crt = this.abortController = new AbortController();
151
156
  fetchEventSource(requestInit.request, {
152
157
  method: requestInit.method,
153
- signal: this.abortController.signal,
158
+ signal: crt.signal,
154
159
  headers: requestInit.headers,
155
160
  keepalive: true,
156
161
  openWhenHidden: false,
@@ -176,7 +181,7 @@ export class ReactiveFetchEventSource {
176
181
  else {
177
182
  this.log.error('Failed to open Event Source due to Error ' + response.status, response);
178
183
  this.updateState(ReactiveEventSourceState.ERROR);
179
- this.tryReconnect(); // TODO Necessary if keep alive = true?
184
+ this.tryReconnect();
180
185
  }
181
186
  }
182
187
  handleOnError(err) {
@@ -185,7 +190,7 @@ export class ReactiveFetchEventSource {
185
190
  if (!this.closed) {
186
191
  // There was an error - try reconnecting
187
192
  this.log.debug('Attempting to reconnect event-source ...');
188
- this.tryReconnect(); // TODO Necessary if keep alive = true?
193
+ this.tryReconnect();
189
194
  }
190
195
  else {
191
196
  this.log.debug('There was an error in the sse connection (closed).', err);
@@ -193,15 +198,12 @@ export class ReactiveFetchEventSource {
193
198
  }
194
199
  tryReconnect() {
195
200
  if (!this._closed) {
196
- setTimeout(() => this.reconnect(), 3000); // Delay the reconnect
201
+ setTimeout(() => this.reconnect(), this.RECONNECT_DELAY_MS); // Delay the reconnect
197
202
  }
198
203
  }
199
204
  handleOnMessage(msg) {
200
205
  const eventType = this.eventTypeOrDefault(msg.event);
201
- const topic = this.eventTopics.get(eventType);
202
- if (topic) {
203
- topic.next(new ReactiveSSeMessage(msg.id, eventType, msg.data, msg.retry));
204
- }
206
+ this.eventRelay.next(new ReactiveSSeMessage(msg.id, eventType, msg.data, msg.retry));
205
207
  }
206
208
  handleOnClose() {
207
209
  this.log.debug('EventSource has closed.');
@@ -211,7 +213,7 @@ export class ReactiveFetchEventSource {
211
213
  this.state$.next(state);
212
214
  }
213
215
  eventTypeOrDefault(type) {
214
- return type ?? 'message';
216
+ return type ?? this.EVENT_TYPE_DEFAULT;
215
217
  }
216
218
  }
217
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"reactive-fetch-event-source.js","sourceRoot":"","sources":["../../../../../../../../projects/elderbyte/ngx-starter/src/lib/features/event-source/fetch/reactive-fetch-event-source.ts"],"names":[],"mappings":"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","sourcesContent":["import {EventSourceMessage, EventStreamContentType, fetchEventSource} from '@microsoft/fetch-event-source';\nimport {LoggerFactory} from '@elderbyte/ts-logger';\nimport {BehaviorSubject, Observable, Subject, Subscription} from 'rxjs';\nimport {map} from 'rxjs/operators';\n\nexport enum ReactiveEventSourceState {\n  IDLE = 'IDLE',\n  CONNECTING = 'CONNECTING',\n  OPEN = 'OPEN',\n  CLOSED = 'CLOSED',\n  ERROR = 'ERROR',\n}\n\nexport class ReactiveSSeMessage<T = string> {\n  constructor(\n    public readonly id: string,\n    public readonly event: string,\n    public readonly data: T,\n    public readonly retry?: number,\n  ) { }\n}\n\nexport interface EventSourceRequestInit {\n  request: RequestInfo,\n  method?: string,\n  headers?: Record<string, string>\n}\n\nexport class ReactiveFetchEventSource<T = any> {\n\n  /***************************************************************************\n   *                                                                         *\n   * Fields                                                                  *\n   *                                                                         *\n   **************************************************************************/\n\n  private readonly log = LoggerFactory.getLogger(this.constructor.name);\n  private _closed: boolean;\n  private readonly abortController = new AbortController();\n\n  private readonly state$ = new BehaviorSubject<ReactiveEventSourceState>(ReactiveEventSourceState.IDLE);\n  private _unsubscribeSub: Subscription;\n\n  private readonly eventTopics = new Map<string, Subject<ReactiveSSeMessage>>();\n\n\n  /***************************************************************************\n   *                                                                         *\n   * Static Builder                                                          *\n   *                                                                         *\n   **************************************************************************/\n\n  public static staticRequest<T>(\n    requestInit: EventSourceRequestInit,\n  ): ReactiveFetchEventSource<T> {\n    if (!requestInit) {\n      throw new Error('You must provide a event source url!');\n    }\n    return new ReactiveFetchEventSource(() => requestInit);\n  }\n\n  public static dynamicRequest<T>(\n    requestInitProvider: () => EventSourceRequestInit\n  ): ReactiveFetchEventSource<T> {\n    return new ReactiveFetchEventSource(requestInitProvider);\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Constructor                                                             *\n   *                                                                         *\n   **************************************************************************/\n\n  private constructor(\n    private readonly requestInitProvider: () => EventSourceRequestInit,\n  ) {\n    if (!requestInitProvider) {\n      throw new Error('You must provide a event source url provider!');\n    }\n    this.open();\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Properties                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  public get closed(): boolean {\n    return this._closed;\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Public API                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  /**\n   * Open the event source.\n   *\n   * Note: The connection is opened automatically upon object creation.\n   * This method should only be used if this reactive-event-source has\n   * been closed explicitly.\n   */\n  public open(): void {\n    this._closed = false;\n    this.reconnect();\n  }\n\n  /**\n   * Get an event stream for the given event type.\n   * @param eventTypeRaw The event type. Defaults to 'message'.\n   */\n  public events(eventTypeRaw?: string): Observable<ReactiveSSeMessage> {\n    const eventType = this.eventTypeOrDefault(eventTypeRaw);\n    if (!this.eventTopics.has(eventType)) {\n      this.eventTopics.set(eventType, new Subject<ReactiveSSeMessage>());\n    }\n    const eventSubj = this.eventTopics.get(eventType);\n    return eventSubj.asObservable();\n  }\n\n  /**\n   * Get an event stream of messages parsed from json.\n   * (event.data must be in json format)\n   *\n   * @param eventType The event type. Defaults to 'message'.\n   */\n  public eventsJson(eventType?: string): Observable<T> {\n    return this.events(eventType).pipe(\n      map(event => JSON.parse(event.data))\n    );\n  }\n\n  /**\n   * Close this event source. It won't reconnect.\n   */\n  public close(): void {\n    this._closed = true;\n    this.closeCurrent();\n    this.log.debug('Closing the event-source.');\n  }\n\n  /**\n   * Keeps this reactive event source open until the given observable emits an event.\n   * @param unsubscribe\n   */\n  public openUntil(unsubscribe: Observable<any>): this {\n    this.cleanUpOpenUntil();\n    this._unsubscribeSub = unsubscribe\n      .subscribe(\n        () => this.close()\n      );\n    return this;\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Private methods                                                         *\n   *                                                                         *\n   **************************************************************************/\n\n  private cleanUpOpenUntil(): void {\n    if (this._unsubscribeSub) {\n      this._unsubscribeSub.unsubscribe();\n      this._unsubscribeSub = null;\n    }\n  }\n\n  /**\n   * Close the current event source\n   */\n  private closeCurrent(reason?: string): void {\n    this.log.debug('Closing the event-source, reason: ' + reason);\n    this.abortController.abort(reason);\n  }\n\n  /**\n   * Reconnects this event source, unless closed is true;\n   */\n  private reconnect(): void {\n    if (this._closed) {\n      return;\n    }\n    this.updateState(ReactiveEventSourceState.CONNECTING);\n    this.closeCurrent('Reconnect');\n    this.openEventSource();\n  }\n\n  private openEventSource(): void {\n    const requestInit = this.requestInitProvider();\n    fetchEventSource(requestInit.request, {\n      method: requestInit.method,\n      signal: this.abortController.signal,\n      headers: requestInit.headers,\n      keepalive: true,\n      openWhenHidden: false,\n      onopen: response => this.handleOnOpen(response),\n      onmessage: msg => this.handleOnMessage(msg),\n      onerror: err => this.handleOnError(err),\n      onclose: () => this.handleOnClose()\n    }).then(r => {\n\n    })\n  }\n\n  private handleOnOpen(response: Response): Promise<void> {\n    if (response.ok && response.headers.get('content-type') === EventStreamContentType) {\n      this.log.debug('EventSource connection sucessfully opened to: ' + response.url +\n        ', state: ' + response.status, response);\n      this.updateState(ReactiveEventSourceState.OPEN);\n      return; // everything's good\n    } else if (response.status >= 400 && response.status < 500 && response.status !== 429) {\n      // client-side errors are usually non-retriable:\n      this.updateState(ReactiveEventSourceState.ERROR);\n      this.log.error('Failed to open Event Source due to Client Error ' + response.status, response);\n    } else {\n      this.log.error('Failed to open Event Source due to Error ' + response.status, response);\n      this.updateState(ReactiveEventSourceState.ERROR);\n      this.tryReconnect(); // TODO Necessary if keep alive = true?\n    }\n  }\n\n  private handleOnError(err: any): void {\n    this.updateState(ReactiveEventSourceState.ERROR);\n    this.log.trace('There was an SSE error.', err);\n    if (!this.closed) {\n      // There was an error - try reconnecting\n      this.log.debug('Attempting to reconnect event-source ...');\n      this.tryReconnect(); // TODO Necessary if keep alive = true?\n    } else {\n      this.log.debug('There was an error in the sse connection (closed).', err);\n    }\n  }\n\n  private tryReconnect(): void {\n    if (!this._closed) {\n      setTimeout(() => this.reconnect(), 3000); // Delay the reconnect\n    }\n  }\n\n  private handleOnMessage(msg: EventSourceMessage): void {\n    const eventType = this.eventTypeOrDefault(msg.event);\n    const topic = this.eventTopics.get(eventType);\n    if (topic) {\n      topic.next(new ReactiveSSeMessage(\n        msg.id,\n        eventType,\n        msg.data,\n        msg.retry\n      ))\n    }\n  }\n\n  private handleOnClose(): void {\n    this.log.debug('EventSource has closed.')\n    this.updateState(ReactiveEventSourceState.CLOSED);\n  }\n\n  private updateState(state: ReactiveEventSourceState) {\n    this.state$.next(state);\n  }\n\n  private eventTypeOrDefault(type?: string): string {\n    return type ?? 'message';\n  }\n}\n"]}
219
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"reactive-fetch-event-source.js","sourceRoot":"","sources":["../../../../../../../../projects/elderbyte/ngx-starter/src/lib/features/event-source/fetch/reactive-fetch-event-source.ts"],"names":[],"mappings":"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","sourcesContent":["import {EventSourceMessage, EventStreamContentType, fetchEventSource} from '@microsoft/fetch-event-source';\nimport {LoggerFactory} from '@elderbyte/ts-logger';\nimport {BehaviorSubject, Observable, Subject, Subscription} from 'rxjs';\nimport {filter, map} from 'rxjs/operators';\n\nexport enum ReactiveEventSourceState {\n  IDLE = 'IDLE',\n  CONNECTING = 'CONNECTING',\n  OPEN = 'OPEN',\n  CLOSED = 'CLOSED',\n  ERROR = 'ERROR',\n}\n\nexport class ReactiveSSeMessage<T = any> {\n  constructor(\n    public readonly id: string,\n    public readonly event: string,\n    public readonly dataRaw: string,\n    public readonly retry?: number,\n  ) {\n  }\n\n  public json(): T {\n    return JSON.parse(this.dataRaw);\n  }\n\n  public jsonAs<TCustom>(): TCustom {\n    return this.json() as unknown as TCustom;\n  }\n}\n\nexport interface EventSourceRequestInit {\n  request: RequestInfo,\n  method?: string,\n  headers?: Record<string, string>\n}\n\nexport class ReactiveFetchEventSource<T = any> {\n\n  /***************************************************************************\n   *                                                                         *\n   * Fields                                                                  *\n   *                                                                         *\n   **************************************************************************/\n\n  private readonly EVENT_TYPE_DEFAULT = 'message';\n  private readonly RECONNECT_DELAY_MS = 3000;\n\n  private readonly log = LoggerFactory.getLogger(this.constructor.name);\n  private _closed: boolean;\n  private abortController: AbortController = null;\n\n  private readonly state$ = new BehaviorSubject<ReactiveEventSourceState>(ReactiveEventSourceState.IDLE);\n  private _unsubscribeSub: Subscription;\n\n  private readonly eventRelay = new Subject<ReactiveSSeMessage<T>>();\n\n  /***************************************************************************\n   *                                                                         *\n   * Static Builder                                                          *\n   *                                                                         *\n   **************************************************************************/\n\n  public static staticRequest<T>(\n    requestInit: EventSourceRequestInit,\n  ): ReactiveFetchEventSource<T> {\n    if (!requestInit) {\n      throw new Error('You must provide a event source url!');\n    }\n    return new ReactiveFetchEventSource(() => requestInit);\n  }\n\n  public static dynamicRequest<T>(\n    requestInitProvider: () => EventSourceRequestInit\n  ): ReactiveFetchEventSource<T> {\n    return new ReactiveFetchEventSource(requestInitProvider);\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Constructor                                                             *\n   *                                                                         *\n   **************************************************************************/\n\n  private constructor(\n    private readonly requestInitProvider: () => EventSourceRequestInit,\n  ) {\n    if (!requestInitProvider) {\n      throw new Error('You must provide a event source url provider!');\n    }\n    this.open();\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Properties                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  public get closed(): boolean {\n    return this._closed;\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Public API                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  /**\n   * Open the event source.\n   *\n   * Note: The connection is opened automatically upon object creation.\n   * This method should only be used if this reactive-event-source has\n   * been closed explicitly.\n   */\n  public open(): void {\n    this._closed = false;\n    this.reconnect();\n  }\n\n  /**\n   * Get an event stream for the given event type.\n   * @param eventTypeRaw The event type. Defaults to 'message'.\n   */\n  public streamEventsOfType(eventTypeRaw?: string): Observable<ReactiveSSeMessage<T>> {\n    const eventType = this.eventTypeOrDefault(eventTypeRaw);\n    return this.streamEvents().pipe(\n      filter(e => e.event == eventType)\n    );\n  }\n\n  /**\n   * Get an event stream for all events no matter their type.\n   */\n  public streamEvents(): Observable<ReactiveSSeMessage<T>> {\n    return this.eventRelay.asObservable();\n  }\n\n  /**\n   * Close this event source. It won't reconnect.\n   */\n  public close(): void {\n    this._closed = true;\n    this.closeCurrent();\n    this.log.debug('Closing the event-source.');\n  }\n\n  /**\n   * Keeps this reactive event source open until the given observable emits an event.\n   * @param unsubscribe\n   */\n  public openUntil(unsubscribe: Observable<any>): this {\n    this.cleanUpOpenUntil();\n    this._unsubscribeSub = unsubscribe\n      .subscribe(\n        () => this.close()\n      );\n    return this;\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Private methods                                                         *\n   *                                                                         *\n   **************************************************************************/\n\n  private cleanUpOpenUntil(): void {\n    if (this._unsubscribeSub) {\n      this._unsubscribeSub.unsubscribe();\n      this._unsubscribeSub = null;\n    }\n  }\n\n  /**\n   * Close the current event source\n   */\n  private closeCurrent(reason?: string): void {\n    this.log.debug('Closing the event-source, reason: ' + reason);\n    const crt = this.abortController;\n    if (crt) {\n      crt.abort(reason);\n    }\n  }\n\n  /**\n   * Reconnects this event source, unless closed is true;\n   */\n  private reconnect(): void {\n    if (this._closed) {\n      return;\n    }\n    this.updateState(ReactiveEventSourceState.CONNECTING);\n    this.closeCurrent('Reconnect');\n    this.openEventSource();\n  }\n\n  private openEventSource(): void {\n    const requestInit = this.requestInitProvider();\n    const crt = this.abortController = new AbortController();\n    fetchEventSource(requestInit.request, {\n      method: requestInit.method,\n      signal: crt.signal,\n      headers: requestInit.headers,\n      keepalive: true,\n      openWhenHidden: false,\n      onopen: response => this.handleOnOpen(response),\n      onmessage: msg => this.handleOnMessage(msg),\n      onerror: err => this.handleOnError(err),\n      onclose: () => this.handleOnClose()\n    }).then(r => {\n\n    });\n  }\n\n  private handleOnOpen(response: Response): Promise<void> {\n    if (response.ok && response.headers.get('content-type') === EventStreamContentType) {\n      this.log.debug('EventSource connection sucessfully opened to: ' + response.url +\n        ', state: ' + response.status, response);\n      this.updateState(ReactiveEventSourceState.OPEN);\n      return; // everything's good\n    } else if (response.status >= 400 && response.status < 500 && response.status !== 429) {\n      // client-side errors are usually non-retriable:\n      this.updateState(ReactiveEventSourceState.ERROR);\n      this.log.error('Failed to open Event Source due to Client Error ' + response.status, response);\n    } else {\n      this.log.error('Failed to open Event Source due to Error ' + response.status, response);\n      this.updateState(ReactiveEventSourceState.ERROR);\n      this.tryReconnect();\n    }\n  }\n\n  private handleOnError(err: any): void {\n    this.updateState(ReactiveEventSourceState.ERROR);\n    this.log.trace('There was an SSE error.', err);\n    if (!this.closed) {\n      // There was an error - try reconnecting\n      this.log.debug('Attempting to reconnect event-source ...');\n      this.tryReconnect();\n    } else {\n      this.log.debug('There was an error in the sse connection (closed).', err);\n    }\n  }\n\n  private tryReconnect(): void {\n    if (!this._closed) {\n      setTimeout(() => this.reconnect(), this.RECONNECT_DELAY_MS); // Delay the reconnect\n    }\n  }\n\n  private handleOnMessage(msg: EventSourceMessage): void {\n    const eventType = this.eventTypeOrDefault(msg.event);\n    this.eventRelay.next(\n      new ReactiveSSeMessage<T>(\n        msg.id,\n        eventType,\n        msg.data,\n        msg.retry\n      )\n    );\n  }\n\n  private handleOnClose(): void {\n    this.log.debug('EventSource has closed.');\n    this.updateState(ReactiveEventSourceState.CLOSED);\n  }\n\n  private updateState(state: ReactiveEventSourceState) {\n    this.state$.next(state);\n  }\n\n  private eventTypeOrDefault(type?: string): string {\n    return type ?? this.EVENT_TYPE_DEFAULT;\n  }\n}\n"]}
@@ -31552,12 +31552,18 @@ var ReactiveEventSourceState;
31552
31552
  ReactiveEventSourceState["ERROR"] = "ERROR";
31553
31553
  })(ReactiveEventSourceState || (ReactiveEventSourceState = {}));
31554
31554
  class ReactiveSSeMessage {
31555
- constructor(id, event, data, retry) {
31555
+ constructor(id, event, dataRaw, retry) {
31556
31556
  this.id = id;
31557
31557
  this.event = event;
31558
- this.data = data;
31558
+ this.dataRaw = dataRaw;
31559
31559
  this.retry = retry;
31560
31560
  }
31561
+ json() {
31562
+ return JSON.parse(this.dataRaw);
31563
+ }
31564
+ jsonAs() {
31565
+ return this.json();
31566
+ }
31561
31567
  }
31562
31568
  class ReactiveFetchEventSource {
31563
31569
  /***************************************************************************
@@ -31586,10 +31592,12 @@ class ReactiveFetchEventSource {
31586
31592
  * Fields *
31587
31593
  * *
31588
31594
  **************************************************************************/
31595
+ this.EVENT_TYPE_DEFAULT = 'message';
31596
+ this.RECONNECT_DELAY_MS = 3000;
31589
31597
  this.log = LoggerFactory.getLogger(this.constructor.name);
31590
- this.abortController = new AbortController();
31598
+ this.abortController = null;
31591
31599
  this.state$ = new BehaviorSubject(ReactiveEventSourceState.IDLE);
31592
- this.eventTopics = new Map();
31600
+ this.eventRelay = new Subject();
31593
31601
  if (!requestInitProvider) {
31594
31602
  throw new Error('You must provide a event source url provider!');
31595
31603
  }
@@ -31623,22 +31631,15 @@ class ReactiveFetchEventSource {
31623
31631
  * Get an event stream for the given event type.
31624
31632
  * @param eventTypeRaw The event type. Defaults to 'message'.
31625
31633
  */
31626
- events(eventTypeRaw) {
31634
+ streamEventsOfType(eventTypeRaw) {
31627
31635
  const eventType = this.eventTypeOrDefault(eventTypeRaw);
31628
- if (!this.eventTopics.has(eventType)) {
31629
- this.eventTopics.set(eventType, new Subject());
31630
- }
31631
- const eventSubj = this.eventTopics.get(eventType);
31632
- return eventSubj.asObservable();
31636
+ return this.streamEvents().pipe(filter(e => e.event == eventType));
31633
31637
  }
31634
31638
  /**
31635
- * Get an event stream of messages parsed from json.
31636
- * (event.data must be in json format)
31637
- *
31638
- * @param eventType The event type. Defaults to 'message'.
31639
+ * Get an event stream for all events no matter their type.
31639
31640
  */
31640
- eventsJson(eventType) {
31641
- return this.events(eventType).pipe(map(event => JSON.parse(event.data)));
31641
+ streamEvents() {
31642
+ return this.eventRelay.asObservable();
31642
31643
  }
31643
31644
  /**
31644
31645
  * Close this event source. It won't reconnect.
@@ -31674,7 +31675,10 @@ class ReactiveFetchEventSource {
31674
31675
  */
31675
31676
  closeCurrent(reason) {
31676
31677
  this.log.debug('Closing the event-source, reason: ' + reason);
31677
- this.abortController.abort(reason);
31678
+ const crt = this.abortController;
31679
+ if (crt) {
31680
+ crt.abort(reason);
31681
+ }
31678
31682
  }
31679
31683
  /**
31680
31684
  * Reconnects this event source, unless closed is true;
@@ -31689,9 +31693,10 @@ class ReactiveFetchEventSource {
31689
31693
  }
31690
31694
  openEventSource() {
31691
31695
  const requestInit = this.requestInitProvider();
31696
+ const crt = this.abortController = new AbortController();
31692
31697
  fetchEventSource(requestInit.request, {
31693
31698
  method: requestInit.method,
31694
- signal: this.abortController.signal,
31699
+ signal: crt.signal,
31695
31700
  headers: requestInit.headers,
31696
31701
  keepalive: true,
31697
31702
  openWhenHidden: false,
@@ -31717,7 +31722,7 @@ class ReactiveFetchEventSource {
31717
31722
  else {
31718
31723
  this.log.error('Failed to open Event Source due to Error ' + response.status, response);
31719
31724
  this.updateState(ReactiveEventSourceState.ERROR);
31720
- this.tryReconnect(); // TODO Necessary if keep alive = true?
31725
+ this.tryReconnect();
31721
31726
  }
31722
31727
  }
31723
31728
  handleOnError(err) {
@@ -31726,7 +31731,7 @@ class ReactiveFetchEventSource {
31726
31731
  if (!this.closed) {
31727
31732
  // There was an error - try reconnecting
31728
31733
  this.log.debug('Attempting to reconnect event-source ...');
31729
- this.tryReconnect(); // TODO Necessary if keep alive = true?
31734
+ this.tryReconnect();
31730
31735
  }
31731
31736
  else {
31732
31737
  this.log.debug('There was an error in the sse connection (closed).', err);
@@ -31734,15 +31739,12 @@ class ReactiveFetchEventSource {
31734
31739
  }
31735
31740
  tryReconnect() {
31736
31741
  if (!this._closed) {
31737
- setTimeout(() => this.reconnect(), 3000); // Delay the reconnect
31742
+ setTimeout(() => this.reconnect(), this.RECONNECT_DELAY_MS); // Delay the reconnect
31738
31743
  }
31739
31744
  }
31740
31745
  handleOnMessage(msg) {
31741
31746
  const eventType = this.eventTypeOrDefault(msg.event);
31742
- const topic = this.eventTopics.get(eventType);
31743
- if (topic) {
31744
- topic.next(new ReactiveSSeMessage(msg.id, eventType, msg.data, msg.retry));
31745
- }
31747
+ this.eventRelay.next(new ReactiveSSeMessage(msg.id, eventType, msg.data, msg.retry));
31746
31748
  }
31747
31749
  handleOnClose() {
31748
31750
  this.log.debug('EventSource has closed.');
@@ -31752,7 +31754,7 @@ class ReactiveFetchEventSource {
31752
31754
  this.state$.next(state);
31753
31755
  }
31754
31756
  eventTypeOrDefault(type) {
31755
- return type ?? 'message';
31757
+ return type ?? this.EVENT_TYPE_DEFAULT;
31756
31758
  }
31757
31759
  }
31758
31760