@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 {
|
|
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,
|
|
14
|
+
constructor(id, event, dataRaw, retry) {
|
|
15
15
|
this.id = id;
|
|
16
16
|
this.event = event;
|
|
17
|
-
this.
|
|
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 =
|
|
57
|
+
this.abortController = null;
|
|
50
58
|
this.state$ = new BehaviorSubject(ReactiveEventSourceState.IDLE);
|
|
51
|
-
this.
|
|
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
|
-
|
|
93
|
+
streamEventsOfType(eventTypeRaw) {
|
|
86
94
|
const eventType = this.eventTypeOrDefault(eventTypeRaw);
|
|
87
|
-
|
|
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
|
|
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
|
-
|
|
100
|
-
return this.
|
|
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
|
|
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:
|
|
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();
|
|
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();
|
|
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(),
|
|
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
|
-
|
|
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 ??
|
|
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,
|
|
31555
|
+
constructor(id, event, dataRaw, retry) {
|
|
31556
31556
|
this.id = id;
|
|
31557
31557
|
this.event = event;
|
|
31558
|
-
this.
|
|
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 =
|
|
31598
|
+
this.abortController = null;
|
|
31591
31599
|
this.state$ = new BehaviorSubject(ReactiveEventSourceState.IDLE);
|
|
31592
|
-
this.
|
|
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
|
-
|
|
31634
|
+
streamEventsOfType(eventTypeRaw) {
|
|
31627
31635
|
const eventType = this.eventTypeOrDefault(eventTypeRaw);
|
|
31628
|
-
|
|
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
|
|
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
|
-
|
|
31641
|
-
return this.
|
|
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
|
|
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:
|
|
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();
|
|
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();
|
|
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(),
|
|
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
|
-
|
|
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 ??
|
|
31757
|
+
return type ?? this.EVENT_TYPE_DEFAULT;
|
|
31756
31758
|
}
|
|
31757
31759
|
}
|
|
31758
31760
|
|