@elderbyte/ngx-starter 17.6.3 → 17.7.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/esm2022/lib/common/data/field-comparator.mjs +11 -2
- package/esm2022/lib/features/event-source/fetch/public_api.mjs +3 -0
- package/esm2022/lib/features/event-source/fetch/reactive-fetch-event-source.mjs +212 -0
- package/esm2022/lib/features/event-source/fetch/reactive-fetch-event-source.service.mjs +53 -0
- package/esm2022/lib/features/event-source/public_api.mjs +3 -3
- package/esm2022/lib/features/event-source/standard/elder-event-source.service.mjs +56 -0
- package/esm2022/lib/features/event-source/standard/public_api.mjs +3 -0
- package/esm2022/lib/features/event-source/standard/reactive-event-source.mjs +198 -0
- package/esm2022/lib/features/kafent/sse/kafent-event-stream-sse.service.mjs +2 -2
- package/esm2022/lib/features/kafent/sse/kafent-topic-sse.mjs +1 -1
- package/fesm2022/elderbyte-ngx-starter.mjs +271 -4
- package/fesm2022/elderbyte-ngx-starter.mjs.map +1 -1
- package/lib/features/event-source/fetch/public_api.d.ts +2 -0
- package/lib/features/event-source/fetch/reactive-fetch-event-source.d.ts +108 -0
- package/lib/features/event-source/fetch/reactive-fetch-event-source.service.d.ts +39 -0
- package/lib/features/event-source/public_api.d.ts +2 -2
- package/lib/features/event-source/standard/public_api.d.ts +2 -0
- package/lib/features/kafent/sse/kafent-event-stream-sse.service.d.ts +1 -1
- package/lib/features/kafent/sse/kafent-topic-sse.d.ts +1 -1
- package/package.json +3 -2
- package/esm2022/lib/features/event-source/elder-event-source.service.mjs +0 -56
- package/esm2022/lib/features/event-source/reactive-event-source.mjs +0 -198
- /package/lib/features/event-source/{elder-event-source.service.d.ts → standard/elder-event-source.service.d.ts} +0 -0
- /package/lib/features/event-source/{reactive-event-source.d.ts → standard/reactive-event-source.d.ts} +0 -0
|
@@ -0,0 +1,198 @@
|
|
|
1
|
+
import { Subject } from 'rxjs';
|
|
2
|
+
import { LoggerFactory } from '@elderbyte/ts-logger';
|
|
3
|
+
import { map } from 'rxjs/operators';
|
|
4
|
+
/**
|
|
5
|
+
* This class provides a reactive wrapper around an event source.
|
|
6
|
+
*
|
|
7
|
+
* It supports event type filtering out of the box and manages the registrations.
|
|
8
|
+
*
|
|
9
|
+
* Additionally, it also handles reconnects since the default reconect handling by browsers
|
|
10
|
+
* is not working in a lot of cases (5xx errors etc).
|
|
11
|
+
*
|
|
12
|
+
*/
|
|
13
|
+
export class ReactiveEventSource {
|
|
14
|
+
/***************************************************************************
|
|
15
|
+
* *
|
|
16
|
+
* Static Builder *
|
|
17
|
+
* *
|
|
18
|
+
**************************************************************************/
|
|
19
|
+
static staticUrl(zone, eventSourceUrl, eventSourceInitDict) {
|
|
20
|
+
if (!eventSourceUrl) {
|
|
21
|
+
throw new Error('You must provide a event source url!');
|
|
22
|
+
}
|
|
23
|
+
return new ReactiveEventSource(zone, () => eventSourceUrl, eventSourceInitDict);
|
|
24
|
+
}
|
|
25
|
+
static dynamicUrl(zone, eventSourceUrlProvider, eventSourceInitDict) {
|
|
26
|
+
return new ReactiveEventSource(zone, eventSourceUrlProvider, eventSourceInitDict);
|
|
27
|
+
}
|
|
28
|
+
/***************************************************************************
|
|
29
|
+
* *
|
|
30
|
+
* Constructor *
|
|
31
|
+
* *
|
|
32
|
+
**************************************************************************/
|
|
33
|
+
constructor(zone, eventSourceUrlProvider, eventSourceInitDict) {
|
|
34
|
+
this.zone = zone;
|
|
35
|
+
this.eventSourceUrlProvider = eventSourceUrlProvider;
|
|
36
|
+
this.eventSourceInitDict = eventSourceInitDict;
|
|
37
|
+
/***************************************************************************
|
|
38
|
+
* *
|
|
39
|
+
* Fields *
|
|
40
|
+
* *
|
|
41
|
+
**************************************************************************/
|
|
42
|
+
this.logger = LoggerFactory.getLogger(this.constructor.name);
|
|
43
|
+
this.eventTopics = new Map();
|
|
44
|
+
this.currentListeners = new Set();
|
|
45
|
+
if (!eventSourceUrlProvider) {
|
|
46
|
+
throw new Error('You must provide a event source url provider!');
|
|
47
|
+
}
|
|
48
|
+
this.open();
|
|
49
|
+
}
|
|
50
|
+
/***************************************************************************
|
|
51
|
+
* *
|
|
52
|
+
* Properties *
|
|
53
|
+
* *
|
|
54
|
+
**************************************************************************/
|
|
55
|
+
get closed() {
|
|
56
|
+
return this._closed;
|
|
57
|
+
}
|
|
58
|
+
/***************************************************************************
|
|
59
|
+
* *
|
|
60
|
+
* Public API *
|
|
61
|
+
* *
|
|
62
|
+
**************************************************************************/
|
|
63
|
+
/**
|
|
64
|
+
* Open the event source.
|
|
65
|
+
*
|
|
66
|
+
* Note: The connection is opened automatically upon object creation.
|
|
67
|
+
* This method should only be used if this reactive-event-source has
|
|
68
|
+
* been closed explicitly.
|
|
69
|
+
*/
|
|
70
|
+
open() {
|
|
71
|
+
this._closed = false;
|
|
72
|
+
this.reconnect();
|
|
73
|
+
}
|
|
74
|
+
/**
|
|
75
|
+
* Get an event stream for the given event type.
|
|
76
|
+
* @param eventType The event type. Defaults to 'message'.
|
|
77
|
+
*/
|
|
78
|
+
events(eventType = 'message') {
|
|
79
|
+
if (!this.eventTopics.has(eventType)) {
|
|
80
|
+
this.eventTopics.set(eventType, new Subject());
|
|
81
|
+
this.ensureRegistrations();
|
|
82
|
+
}
|
|
83
|
+
const eventSubj = this.eventTopics.get(eventType);
|
|
84
|
+
return eventSubj.asObservable();
|
|
85
|
+
}
|
|
86
|
+
/**
|
|
87
|
+
* Get an event stream of messages parsed from json.
|
|
88
|
+
* (event.data must be in json format)
|
|
89
|
+
*
|
|
90
|
+
* @param eventType The event type. Defaults to 'message'.
|
|
91
|
+
*/
|
|
92
|
+
eventsJson(eventType = 'message') {
|
|
93
|
+
return this.events(eventType).pipe(map(event => JSON.parse(event.data)));
|
|
94
|
+
}
|
|
95
|
+
/**
|
|
96
|
+
* Close this event source. It wont reconnect.
|
|
97
|
+
*/
|
|
98
|
+
close() {
|
|
99
|
+
this._closed = true;
|
|
100
|
+
this.closeCurrent();
|
|
101
|
+
this.logger.debug('Closing the event-source.');
|
|
102
|
+
}
|
|
103
|
+
/**
|
|
104
|
+
* Keeps this reactive event source open until the given observable emits an event.
|
|
105
|
+
* @param unsubscribe
|
|
106
|
+
*/
|
|
107
|
+
openUntil(unsubscribe) {
|
|
108
|
+
this.cleanUpOpenUntil();
|
|
109
|
+
this._unsubscribeSub = unsubscribe
|
|
110
|
+
.subscribe(() => this.close());
|
|
111
|
+
return this;
|
|
112
|
+
}
|
|
113
|
+
/***************************************************************************
|
|
114
|
+
* *
|
|
115
|
+
* Private methods *
|
|
116
|
+
* *
|
|
117
|
+
**************************************************************************/
|
|
118
|
+
cleanUpOpenUntil() {
|
|
119
|
+
if (this._unsubscribeSub) {
|
|
120
|
+
this._unsubscribeSub.unsubscribe();
|
|
121
|
+
this._unsubscribeSub = null;
|
|
122
|
+
}
|
|
123
|
+
}
|
|
124
|
+
/**
|
|
125
|
+
* Reconnects this event source, unless closed is true;
|
|
126
|
+
*/
|
|
127
|
+
reconnect() {
|
|
128
|
+
if (this._closed) {
|
|
129
|
+
return;
|
|
130
|
+
}
|
|
131
|
+
this.closeCurrent();
|
|
132
|
+
const eventSourceUrl = this.eventSourceUrlProvider();
|
|
133
|
+
try {
|
|
134
|
+
this.currentSource = new EventSource(eventSourceUrl, this.eventSourceInitDict);
|
|
135
|
+
this.ensureRegistrations();
|
|
136
|
+
this.currentSource.onopen = (event) => {
|
|
137
|
+
this.logger.debug('EventSource connection opened to: ' + eventSourceUrl +
|
|
138
|
+
', state: ' + this.readyStateAsString(this.currentSource), event);
|
|
139
|
+
};
|
|
140
|
+
this.currentSource.onerror = (error) => {
|
|
141
|
+
this.logger.trace('There was an SSE error, current state: ' + this.readyStateAsString(this.currentSource), error);
|
|
142
|
+
if (!this.closed) {
|
|
143
|
+
// There was an error - try reconnecting
|
|
144
|
+
this.logger.debug('Attempting to reconnect event-source at' + eventSourceUrl + '...');
|
|
145
|
+
setTimeout(() => this.reconnect(), 3000); // Delay the reconnect
|
|
146
|
+
}
|
|
147
|
+
else {
|
|
148
|
+
this.logger.debug('There was an error in the sse connection (closed).', error);
|
|
149
|
+
}
|
|
150
|
+
};
|
|
151
|
+
}
|
|
152
|
+
catch (err) {
|
|
153
|
+
this.logger.error('Failed to create EventSource for ' + eventSourceUrl, err);
|
|
154
|
+
setTimeout(() => this.reconnect(), 3000); // Delay the reconnect
|
|
155
|
+
}
|
|
156
|
+
}
|
|
157
|
+
ensureRegistrations() {
|
|
158
|
+
if (this.currentSource) {
|
|
159
|
+
this.eventTopics.forEach((subject, type) => {
|
|
160
|
+
if (!this.currentListeners.has(type)) {
|
|
161
|
+
this.currentSource.addEventListener(type, msg => subject.next(msg), false);
|
|
162
|
+
this.currentListeners.add(type);
|
|
163
|
+
this.logger.debug('Listening to event type: ' + type);
|
|
164
|
+
}
|
|
165
|
+
});
|
|
166
|
+
}
|
|
167
|
+
}
|
|
168
|
+
/**
|
|
169
|
+
* Close the current event source
|
|
170
|
+
*/
|
|
171
|
+
closeCurrent() {
|
|
172
|
+
this.currentListeners.clear();
|
|
173
|
+
if (!this.currentClosed) {
|
|
174
|
+
// Close Event Source if not already closed.
|
|
175
|
+
this.logger.debug('Closing the event-source.');
|
|
176
|
+
this.currentSource.close();
|
|
177
|
+
this.currentSource = null;
|
|
178
|
+
}
|
|
179
|
+
}
|
|
180
|
+
get currentClosed() {
|
|
181
|
+
return !this.currentSource || this.currentSource.readyState === this.currentSource.CLOSED;
|
|
182
|
+
}
|
|
183
|
+
readyStateAsString(source) {
|
|
184
|
+
if (source.readyState === source.OPEN) {
|
|
185
|
+
return 'OPEN';
|
|
186
|
+
}
|
|
187
|
+
else if (source.readyState === source.CLOSED) {
|
|
188
|
+
return 'CLOSED';
|
|
189
|
+
}
|
|
190
|
+
else if (source.readyState === source.CONNECTING) {
|
|
191
|
+
return 'CONNECTING';
|
|
192
|
+
}
|
|
193
|
+
else {
|
|
194
|
+
return 'UNKNOWN';
|
|
195
|
+
}
|
|
196
|
+
}
|
|
197
|
+
}
|
|
198
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"reactive-event-source.js","sourceRoot":"","sources":["../../../../../../../../projects/elderbyte/ngx-starter/src/lib/features/event-source/standard/reactive-event-source.ts"],"names":[],"mappings":"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","sourcesContent":["import {Observable, Subject, Subscription} from 'rxjs';\nimport {NgZone} from '@angular/core';\nimport {LoggerFactory} from '@elderbyte/ts-logger';\nimport {map} from 'rxjs/operators';\n\n/**\n * This class provides a reactive wrapper around an event source.\n *\n * It supports event type filtering out of the box and manages the registrations.\n *\n * Additionally, it also handles reconnects since the default reconect handling by browsers\n * is not working in a lot of cases (5xx errors etc).\n *\n */\nexport class ReactiveEventSource<T = any> {\n\n  /***************************************************************************\n   *                                                                         *\n   * Fields                                                                  *\n   *                                                                         *\n   **************************************************************************/\n\n  private readonly logger = LoggerFactory.getLogger(this.constructor.name);\n\n  private readonly eventTopics = new Map<string, Subject<MessageEvent>>();\n\n  private currentSource: EventSource;\n  private currentListeners = new Set<string>();\n\n  private _closed: boolean;\n\n  private _unsubscribeSub: Subscription;\n\n  /***************************************************************************\n   *                                                                         *\n   * Static Builder                                                          *\n   *                                                                         *\n   **************************************************************************/\n\n  public static staticUrl<T>(\n    zone: NgZone,\n    eventSourceUrl: string,\n    eventSourceInitDict?: EventSourceInit\n  ): ReactiveEventSource<T> {\n\n    if (!eventSourceUrl) {\n      throw new Error('You must provide a event source url!');\n    }\n    return new ReactiveEventSource(zone, () => eventSourceUrl, eventSourceInitDict);\n  }\n\n  public static dynamicUrl<T>(\n    zone: NgZone,\n    eventSourceUrlProvider: () => string,\n    eventSourceInitDict?: EventSourceInit\n  ): ReactiveEventSource<T> {\n    return new ReactiveEventSource(zone, eventSourceUrlProvider, eventSourceInitDict);\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Constructor                                                             *\n   *                                                                         *\n   **************************************************************************/\n\n  private constructor(\n    private readonly zone: NgZone,\n    private eventSourceUrlProvider: () => string,\n    private eventSourceInitDict?: EventSourceInit\n  ) {\n    if (!eventSourceUrlProvider) {\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 eventType The event type. Defaults to 'message'.\n   */\n  public events(eventType: string = 'message'): Observable<MessageEvent> {\n    if (!this.eventTopics.has(eventType)) {\n      this.eventTopics.set(eventType, new Subject<MessageEvent>());\n      this.ensureRegistrations();\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 = 'message'): 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 wont reconnect.\n   */\n  public close(): void {\n    this._closed = true;\n    this.closeCurrent();\n    this.logger.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   * Reconnects this event source, unless closed is true;\n   */\n  private reconnect(): void {\n\n    if (this._closed) { return; }\n\n    this.closeCurrent();\n\n    const eventSourceUrl = this.eventSourceUrlProvider();\n\n    try {\n      this.currentSource = new EventSource(eventSourceUrl, this.eventSourceInitDict);\n\n      this.ensureRegistrations();\n\n      this.currentSource.onopen = (event) => {\n        this.logger.debug('EventSource connection opened to: ' + eventSourceUrl +\n          ', state: ' + this.readyStateAsString(this.currentSource), event);\n      };\n\n      this.currentSource.onerror = (error) => {\n\n        this.logger.trace('There was an SSE error, current state: ' + this.readyStateAsString(this.currentSource), error);\n\n        if (!this.closed) {\n          // There was an error - try reconnecting\n          this.logger.debug('Attempting to reconnect event-source at' + eventSourceUrl + '...');\n          setTimeout(() => this.reconnect(), 3000); // Delay the reconnect\n        } else {\n          this.logger.debug('There was an error in the sse connection (closed).', error);\n        }\n      };\n\n    } catch (err) {\n      this.logger.error('Failed to create EventSource for ' + eventSourceUrl, err);\n      setTimeout(() => this.reconnect(), 3000); // Delay the reconnect\n    }\n  }\n\n  private ensureRegistrations(): void {\n    if (this.currentSource) {\n      this.eventTopics.forEach((subject, type) => {\n\n        if (!this.currentListeners.has(type)) {\n          this.currentSource.addEventListener(\n            type,\n            msg => subject.next(msg as MessageEvent),\n            false\n          );\n          this.currentListeners.add(type);\n          this.logger.debug('Listening to event type: ' + type);\n        }\n      });\n    }\n  }\n\n  /**\n   * Close the current event source\n   */\n  private closeCurrent(): void {\n\n    this.currentListeners.clear();\n\n    if (!this.currentClosed) {\n      // Close Event Source if not already closed.\n      this.logger.debug('Closing the event-source.');\n      this.currentSource.close();\n      this.currentSource = null;\n    }\n  }\n\n  private get currentClosed(): boolean {\n    return !this.currentSource || this.currentSource.readyState === this.currentSource.CLOSED;\n  }\n\n  private readyStateAsString(source: EventSource): string {\n    if (source.readyState === source.OPEN) {\n      return 'OPEN';\n    } else if (source.readyState === source.CLOSED) {\n      return 'CLOSED';\n    } else if (source.readyState === source.CONNECTING) {\n      return 'CONNECTING';\n    } else {\n      return 'UNKNOWN';\n    }\n  }\n}\n"]}
|
|
@@ -7,7 +7,7 @@ import { LoggerFactory } from '@elderbyte/ts-logger';
|
|
|
7
7
|
import { KafentEventStream } from '../kafent-event-stream';
|
|
8
8
|
import * as i0 from "@angular/core";
|
|
9
9
|
import * as i1 from "../kafent-config";
|
|
10
|
-
import * as i2 from "../../event-source/elder-event-source.service";
|
|
10
|
+
import * as i2 from "../../event-source/standard/elder-event-source.service";
|
|
11
11
|
import * as i3 from "../access-token-provider";
|
|
12
12
|
export class KafentSseEventChannel {
|
|
13
13
|
constructor(request, eventChannelSub) {
|
|
@@ -125,4 +125,4 @@ export class KafentEventStreamSse extends KafentEventStream {
|
|
|
125
125
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.4", ngImport: i0, type: KafentEventStreamSse, decorators: [{
|
|
126
126
|
type: Injectable
|
|
127
127
|
}], ctorParameters: () => [{ type: i1.KafentConfig }, { type: i2.ElderEventSourceService }, { type: i3.KafentTokenProvider }] });
|
|
128
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"kafent-event-stream-sse.service.js","sourceRoot":"","sources":["../../../../../../../../projects/elderbyte/ngx-starter/src/lib/features/kafent/sse/kafent-event-stream-sse.service.ts"],"names":[],"mappings":"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","sourcesContent":["import {KafentTopicSse} from './kafent-topic-sse';\nimport {KafentEvent} from '../kafent-event';\nimport {Observable} from 'rxjs';\nimport {KafentTokenProvider} from '../access-token-provider';\nimport {KafentConfig, KafentEventTransport} from '../kafent-config';\nimport {ReactiveEventSource} from '../../event-source/reactive-event-source';\nimport {ElderEventSourceService} from '../../event-source/elder-event-source.service';\nimport {UrlBuilder} from '../../../common/utils/url-builder';\nimport {Injectable} from '@angular/core';\nimport {KafentLiveEventRequest} from '../kafent-live-event-request';\nimport {skip} from 'rxjs/operators';\nimport {LoggerFactory} from '@elderbyte/ts-logger';\nimport {KafentEventStream} from '../kafent-event-stream';\n\n\nexport class KafentSseEventChannel {\n\n  public readonly id: string;\n\n  constructor(\n    public readonly request: KafentLiveEventRequest,\n    public readonly eventChannelSub: KafentTopicSse\n  ) {\n    this.id = KafentSseEventChannel.hash(request);\n  }\n\n\n  public static hash(request: KafentLiveEventRequest): string {\n    return JSON.stringify(request);\n  }\n}\n\n\n@Injectable()\nexport class KafentEventStreamSse extends KafentEventStream {\n\n  /***************************************************************************\n   *                                                                         *\n   * Fields                                                                  *\n   *                                                                         *\n   **************************************************************************/\n\n  private readonly logger = LoggerFactory.getLogger('KafentEventServiceSse');\n\n  private readonly activeEventChannels = new Map<string, KafentSseEventChannel>();\n  private kafentSource: ReactiveEventSource<KafentEvent>;\n\n  private readonly authTokenParam: string;\n\n  /***************************************************************************\n   *                                                                         *\n   * Constructor                                                             *\n   *                                                                         *\n   **************************************************************************/\n\n  constructor(\n    private readonly config: KafentConfig,\n    private readonly eventSourceService: ElderEventSourceService,\n    private readonly tokenProvider: KafentTokenProvider\n  ) {\n    super(KafentEventTransport.SSE);\n    this.authTokenParam = this.config.tokenParam || 'access_token';\n\n    this.logger.info('Initialized KafentEventStreamSse with auth-token param: ' + this.authTokenParam);\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Public API                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  public getLiveEventsWith(topic: KafentLiveEventRequest): Observable<KafentEvent> {\n    const kafentChannel = this.kafentEventChannel(topic);\n    return kafentChannel.eventChannelSub.events;\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Private methods                                                         *\n   *                                                                         *\n   **************************************************************************/\n\n  private kafentEventChannel(request: KafentLiveEventRequest): KafentSseEventChannel {\n    const channelId = KafentSseEventChannel.hash(request);\n    if (!this.activeEventChannels.has(channelId)) {\n      this.activeEventChannels.set(channelId, this.createKafentChannel(request));\n    }\n    return this.activeEventChannels.get(channelId);\n  }\n\n  private createKafentChannel(request: KafentLiveEventRequest): KafentSseEventChannel {\n    const ktopic = new KafentTopicSse(request.topic);\n    const channel = new KafentSseEventChannel(request, ktopic);\n\n\n    const sub = ktopic.connectedCount.pipe(\n      skip(1) // Skip initial behaviour-subject value\n    )\n      .subscribe(\n      connected => {\n        if (connected === 1) {\n          // If the first has connected\n          this.updateKafentSseSub();\n        }\n        if (connected === 0) {\n          // If the last has disconnected\n          // Self deconstruct\n          this.activeEventChannels.delete(channel.id);\n          sub.unsubscribe();\n          this.updateKafentSseSub();\n        }\n      }\n    );\n    return channel;\n  }\n\n  private updateKafentSseSub(): void {\n    this.clearSource();\n\n    const channels = Array.from(this.activeEventChannels.values());\n\n    const usedChannels = channels\n                            .filter(c => c.eventChannelSub.anyConnected);\n\n    if (usedChannels.length > 0) {\n      this.kafentSource = this.buildKafentSource(usedChannels);\n      this.connect(this.kafentSource);\n    }\n  }\n\n  private clearSource(): void {\n    if (this.kafentSource) {\n      this.kafentSource.close();\n      this.kafentSource = null;\n    }\n  }\n\n  private buildKafentSource(channels: KafentSseEventChannel[]): ReactiveEventSource<KafentEvent> {\n    return this.eventSourceService\n      .reactiveEventSourceDynamic(() => this.buildEventSourceUrl(channels));\n  }\n\n  private buildEventSourceUrl(channels: KafentSseEventChannel[]): string {\n    const b = UrlBuilder.parse(this.config.kafentUrl);\n    channels\n      .map(c => c.request)\n      .forEach(\n        (request) => b.appendParam('topicJson', btoa(JSON.stringify(request)))\n      );\n\n    const token = this.tokenProvider.getAccessToken();\n    if (token) {\n      this.logger.debug('Appending param: ' + this.authTokenParam + ' with access_token: ' + token.substring(0, 5) + '...');\n      b.appendParam(this.authTokenParam, token);\n    } else {\n      this.logger.warn('No access_token was provided, skipping auth param: ' + this.authTokenParam);\n    }\n\n    return b.build();\n  }\n\n  private connect(kafentSource: ReactiveEventSource<KafentEvent>): void {\n    this.activeEventChannels.forEach( (v, k) => v.eventChannelSub.connect(kafentSource));\n  }\n\n}\n"]}
|
|
128
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"kafent-event-stream-sse.service.js","sourceRoot":"","sources":["../../../../../../../../projects/elderbyte/ngx-starter/src/lib/features/kafent/sse/kafent-event-stream-sse.service.ts"],"names":[],"mappings":"AAAA,OAAO,EAAC,cAAc,EAAC,MAAM,oBAAoB,CAAC;AAIlD,OAAO,EAAe,oBAAoB,EAAC,MAAM,kBAAkB,CAAC;AAGpE,OAAO,EAAC,UAAU,EAAC,MAAM,mCAAmC,CAAC;AAC7D,OAAO,EAAC,UAAU,EAAC,MAAM,eAAe,CAAC;AAEzC,OAAO,EAAC,IAAI,EAAC,MAAM,gBAAgB,CAAC;AACpC,OAAO,EAAC,aAAa,EAAC,MAAM,sBAAsB,CAAC;AACnD,OAAO,EAAC,iBAAiB,EAAC,MAAM,wBAAwB,CAAC;;;;;AAGzD,MAAM,OAAO,qBAAqB;IAIhC,YACkB,OAA+B,EAC/B,eAA+B;QAD/B,YAAO,GAAP,OAAO,CAAwB;QAC/B,oBAAe,GAAf,eAAe,CAAgB;QAE/C,IAAI,CAAC,EAAE,GAAG,qBAAqB,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;IAChD,CAAC;IAGM,MAAM,CAAC,IAAI,CAAC,OAA+B;QAChD,OAAO,IAAI,CAAC,SAAS,CAAC,OAAO,CAAC,CAAC;IACjC,CAAC;CACF;AAID,MAAM,OAAO,oBAAqB,SAAQ,iBAAiB;IAezD;;;;gFAI4E;IAE5E,YACmB,MAAoB,EACpB,kBAA2C,EAC3C,aAAkC;QAEnD,KAAK,CAAC,oBAAoB,CAAC,GAAG,CAAC,CAAC;QAJf,WAAM,GAAN,MAAM,CAAc;QACpB,uBAAkB,GAAlB,kBAAkB,CAAyB;QAC3C,kBAAa,GAAb,aAAa,CAAqB;QAtBrD;;;;oFAI4E;QAE3D,WAAM,GAAG,aAAa,CAAC,SAAS,CAAC,uBAAuB,CAAC,CAAC;QAE1D,wBAAmB,GAAG,IAAI,GAAG,EAAiC,CAAC;QAiB9E,IAAI,CAAC,cAAc,GAAG,IAAI,CAAC,MAAM,CAAC,UAAU,IAAI,cAAc,CAAC;QAE/D,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,0DAA0D,GAAG,IAAI,CAAC,cAAc,CAAC,CAAC;IACrG,CAAC;IAED;;;;gFAI4E;IAErE,iBAAiB,CAAC,KAA6B;QACpD,MAAM,aAAa,GAAG,IAAI,CAAC,kBAAkB,CAAC,KAAK,CAAC,CAAC;QACrD,OAAO,aAAa,CAAC,eAAe,CAAC,MAAM,CAAC;IAC9C,CAAC;IAED;;;;gFAI4E;IAEpE,kBAAkB,CAAC,OAA+B;QACxD,MAAM,SAAS,GAAG,qBAAqB,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;QACtD,IAAI,CAAC,IAAI,CAAC,mBAAmB,CAAC,GAAG,CAAC,SAAS,CAAC,EAAE;YAC5C,IAAI,CAAC,mBAAmB,CAAC,GAAG,CAAC,SAAS,EAAE,IAAI,CAAC,mBAAmB,CAAC,OAAO,CAAC,CAAC,CAAC;SAC5E;QACD,OAAO,IAAI,CAAC,mBAAmB,CAAC,GAAG,CAAC,SAAS,CAAC,CAAC;IACjD,CAAC;IAEO,mBAAmB,CAAC,OAA+B;QACzD,MAAM,MAAM,GAAG,IAAI,cAAc,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC;QACjD,MAAM,OAAO,GAAG,IAAI,qBAAqB,CAAC,OAAO,EAAE,MAAM,CAAC,CAAC;QAG3D,MAAM,GAAG,GAAG,MAAM,CAAC,cAAc,CAAC,IAAI,CACpC,IAAI,CAAC,CAAC,CAAC,CAAC,uCAAuC;SAChD;aACE,SAAS,CACV,SAAS,CAAC,EAAE;YACV,IAAI,SAAS,KAAK,CAAC,EAAE;gBACnB,6BAA6B;gBAC7B,IAAI,CAAC,kBAAkB,EAAE,CAAC;aAC3B;YACD,IAAI,SAAS,KAAK,CAAC,EAAE;gBACnB,+BAA+B;gBAC/B,mBAAmB;gBACnB,IAAI,CAAC,mBAAmB,CAAC,MAAM,CAAC,OAAO,CAAC,EAAE,CAAC,CAAC;gBAC5C,GAAG,CAAC,WAAW,EAAE,CAAC;gBAClB,IAAI,CAAC,kBAAkB,EAAE,CAAC;aAC3B;QACH,CAAC,CACF,CAAC;QACF,OAAO,OAAO,CAAC;IACjB,CAAC;IAEO,kBAAkB;QACxB,IAAI,CAAC,WAAW,EAAE,CAAC;QAEnB,MAAM,QAAQ,GAAG,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,mBAAmB,CAAC,MAAM,EAAE,CAAC,CAAC;QAE/D,MAAM,YAAY,GAAG,QAAQ;aACJ,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,eAAe,CAAC,YAAY,CAAC,CAAC;QAErE,IAAI,YAAY,CAAC,MAAM,GAAG,CAAC,EAAE;YAC3B,IAAI,CAAC,YAAY,GAAG,IAAI,CAAC,iBAAiB,CAAC,YAAY,CAAC,CAAC;YACzD,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC;SACjC;IACH,CAAC;IAEO,WAAW;QACjB,IAAI,IAAI,CAAC,YAAY,EAAE;YACrB,IAAI,CAAC,YAAY,CAAC,KAAK,EAAE,CAAC;YAC1B,IAAI,CAAC,YAAY,GAAG,IAAI,CAAC;SAC1B;IACH,CAAC;IAEO,iBAAiB,CAAC,QAAiC;QACzD,OAAO,IAAI,CAAC,kBAAkB;aAC3B,0BAA0B,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,mBAAmB,CAAC,QAAQ,CAAC,CAAC,CAAC;IAC1E,CAAC;IAEO,mBAAmB,CAAC,QAAiC;QAC3D,MAAM,CAAC,GAAG,UAAU,CAAC,KAAK,CAAC,IAAI,CAAC,MAAM,CAAC,SAAS,CAAC,CAAC;QAClD,QAAQ;aACL,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,OAAO,CAAC;aACnB,OAAO,CACN,CAAC,OAAO,EAAE,EAAE,CAAC,CAAC,CAAC,WAAW,CAAC,WAAW,EAAE,IAAI,CAAC,IAAI,CAAC,SAAS,CAAC,OAAO,CAAC,CAAC,CAAC,CACvE,CAAC;QAEJ,MAAM,KAAK,GAAG,IAAI,CAAC,aAAa,CAAC,cAAc,EAAE,CAAC;QAClD,IAAI,KAAK,EAAE;YACT,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,mBAAmB,GAAG,IAAI,CAAC,cAAc,GAAG,sBAAsB,GAAG,KAAK,CAAC,SAAS,CAAC,CAAC,EAAE,CAAC,CAAC,GAAG,KAAK,CAAC,CAAC;YACtH,CAAC,CAAC,WAAW,CAAC,IAAI,CAAC,cAAc,EAAE,KAAK,CAAC,CAAC;SAC3C;aAAM;YACL,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,qDAAqD,GAAG,IAAI,CAAC,cAAc,CAAC,CAAC;SAC/F;QAED,OAAO,CAAC,CAAC,KAAK,EAAE,CAAC;IACnB,CAAC;IAEO,OAAO,CAAC,YAA8C;QAC5D,IAAI,CAAC,mBAAmB,CAAC,OAAO,CAAE,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,eAAe,CAAC,OAAO,CAAC,YAAY,CAAC,CAAC,CAAC;IACvF,CAAC;8GAlIU,oBAAoB;kHAApB,oBAAoB;;2FAApB,oBAAoB;kBADhC,UAAU","sourcesContent":["import {KafentTopicSse} from './kafent-topic-sse';\nimport {KafentEvent} from '../kafent-event';\nimport {Observable} from 'rxjs';\nimport {KafentTokenProvider} from '../access-token-provider';\nimport {KafentConfig, KafentEventTransport} from '../kafent-config';\nimport {ReactiveEventSource} from '../../event-source/standard/reactive-event-source';\nimport {ElderEventSourceService} from '../../event-source/standard/elder-event-source.service';\nimport {UrlBuilder} from '../../../common/utils/url-builder';\nimport {Injectable} from '@angular/core';\nimport {KafentLiveEventRequest} from '../kafent-live-event-request';\nimport {skip} from 'rxjs/operators';\nimport {LoggerFactory} from '@elderbyte/ts-logger';\nimport {KafentEventStream} from '../kafent-event-stream';\n\n\nexport class KafentSseEventChannel {\n\n  public readonly id: string;\n\n  constructor(\n    public readonly request: KafentLiveEventRequest,\n    public readonly eventChannelSub: KafentTopicSse\n  ) {\n    this.id = KafentSseEventChannel.hash(request);\n  }\n\n\n  public static hash(request: KafentLiveEventRequest): string {\n    return JSON.stringify(request);\n  }\n}\n\n\n@Injectable()\nexport class KafentEventStreamSse extends KafentEventStream {\n\n  /***************************************************************************\n   *                                                                         *\n   * Fields                                                                  *\n   *                                                                         *\n   **************************************************************************/\n\n  private readonly logger = LoggerFactory.getLogger('KafentEventServiceSse');\n\n  private readonly activeEventChannels = new Map<string, KafentSseEventChannel>();\n  private kafentSource: ReactiveEventSource<KafentEvent>;\n\n  private readonly authTokenParam: string;\n\n  /***************************************************************************\n   *                                                                         *\n   * Constructor                                                             *\n   *                                                                         *\n   **************************************************************************/\n\n  constructor(\n    private readonly config: KafentConfig,\n    private readonly eventSourceService: ElderEventSourceService,\n    private readonly tokenProvider: KafentTokenProvider\n  ) {\n    super(KafentEventTransport.SSE);\n    this.authTokenParam = this.config.tokenParam || 'access_token';\n\n    this.logger.info('Initialized KafentEventStreamSse with auth-token param: ' + this.authTokenParam);\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Public API                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  public getLiveEventsWith(topic: KafentLiveEventRequest): Observable<KafentEvent> {\n    const kafentChannel = this.kafentEventChannel(topic);\n    return kafentChannel.eventChannelSub.events;\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Private methods                                                         *\n   *                                                                         *\n   **************************************************************************/\n\n  private kafentEventChannel(request: KafentLiveEventRequest): KafentSseEventChannel {\n    const channelId = KafentSseEventChannel.hash(request);\n    if (!this.activeEventChannels.has(channelId)) {\n      this.activeEventChannels.set(channelId, this.createKafentChannel(request));\n    }\n    return this.activeEventChannels.get(channelId);\n  }\n\n  private createKafentChannel(request: KafentLiveEventRequest): KafentSseEventChannel {\n    const ktopic = new KafentTopicSse(request.topic);\n    const channel = new KafentSseEventChannel(request, ktopic);\n\n\n    const sub = ktopic.connectedCount.pipe(\n      skip(1) // Skip initial behaviour-subject value\n    )\n      .subscribe(\n      connected => {\n        if (connected === 1) {\n          // If the first has connected\n          this.updateKafentSseSub();\n        }\n        if (connected === 0) {\n          // If the last has disconnected\n          // Self deconstruct\n          this.activeEventChannels.delete(channel.id);\n          sub.unsubscribe();\n          this.updateKafentSseSub();\n        }\n      }\n    );\n    return channel;\n  }\n\n  private updateKafentSseSub(): void {\n    this.clearSource();\n\n    const channels = Array.from(this.activeEventChannels.values());\n\n    const usedChannels = channels\n                            .filter(c => c.eventChannelSub.anyConnected);\n\n    if (usedChannels.length > 0) {\n      this.kafentSource = this.buildKafentSource(usedChannels);\n      this.connect(this.kafentSource);\n    }\n  }\n\n  private clearSource(): void {\n    if (this.kafentSource) {\n      this.kafentSource.close();\n      this.kafentSource = null;\n    }\n  }\n\n  private buildKafentSource(channels: KafentSseEventChannel[]): ReactiveEventSource<KafentEvent> {\n    return this.eventSourceService\n      .reactiveEventSourceDynamic(() => this.buildEventSourceUrl(channels));\n  }\n\n  private buildEventSourceUrl(channels: KafentSseEventChannel[]): string {\n    const b = UrlBuilder.parse(this.config.kafentUrl);\n    channels\n      .map(c => c.request)\n      .forEach(\n        (request) => b.appendParam('topicJson', btoa(JSON.stringify(request)))\n      );\n\n    const token = this.tokenProvider.getAccessToken();\n    if (token) {\n      this.logger.debug('Appending param: ' + this.authTokenParam + ' with access_token: ' + token.substring(0, 5) + '...');\n      b.appendParam(this.authTokenParam, token);\n    } else {\n      this.logger.warn('No access_token was provided, skipping auth param: ' + this.authTokenParam);\n    }\n\n    return b.build();\n  }\n\n  private connect(kafentSource: ReactiveEventSource<KafentEvent>): void {\n    this.activeEventChannels.forEach( (v, k) => v.eventChannelSub.connect(kafentSource));\n  }\n\n}\n"]}
|
|
@@ -95,4 +95,4 @@ export class KafentTopicSse {
|
|
|
95
95
|
}
|
|
96
96
|
}
|
|
97
97
|
}
|
|
98
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
98
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"kafent-topic-sse.js","sourceRoot":"","sources":["../../../../../../../../projects/elderbyte/ngx-starter/src/lib/features/kafent/sse/kafent-topic-sse.ts"],"names":[],"mappings":"AACA,OAAO,EAAC,eAAe,EAAE,UAAU,EAAE,OAAO,EAAe,MAAM,MAAM,CAAC;AAExE,OAAO,EAAC,KAAK,EAAC,MAAM,gBAAgB,CAAC;AAGrC,MAAM,OAAO,cAAc;IAmCzB;;;;gFAI4E;IAE5E,YACmB,KAAa;QAAb,UAAK,GAAL,KAAK,CAAQ;QAtBhC;;WAEG;QACc,UAAK,GAAG,IAAI,OAAO,EAAe,CAAC;QAMpD;;;WAGG;QACc,cAAS,GAAG,IAAI,eAAe,CAAC,CAAC,CAAC,CAAC;QAWlD,IAAI,CAAC,MAAM,GAAG,IAAI,UAAU,CAAC,CAAC,QAAQ,EAAE,EAAE;YAExC,MAAM,GAAG,GAAG,IAAI,CAAC,KAAK,CAAC,SAAS,CAAC,QAAQ,CAAC,CAAC;YAE3C,IAAI,IAAI,CAAC,aAAa,EAAE;gBACtB,IAAI,CAAC,kBAAkB,EAAE,CAAC;gBAC1B,IAAI,CAAC,WAAW,EAAE,CAAC;aACpB;YAED,WAAW;YACX,OAAO,GAAG,EAAE;gBACV,GAAG,CAAC,WAAW,EAAE,CAAC;gBAClB,IAAI,CAAC,kBAAkB,EAAE,CAAC;gBAC1B,IAAI,IAAI,CAAC,aAAa,EAAE;oBACtB,IAAI,CAAC,QAAQ,EAAE,CAAC;iBACjB;YACH,CAAC,CAAC;QACJ,CAAC,CAAC,CAAC;QACH,IAAI,CAAC,GAAG,GAAG,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,KAAK,EAAE,CAAC,CAAC;IACvC,CAAC;IAED;;;;gFAI4E;IAE5E;;;OAGG;IACH,IAAW,MAAM;QACf,OAAO,IAAI,CAAC,GAAG,CAAC;IAClB,CAAC;IAED;;;OAGG;IACH,IAAW,cAAc;QACvB,OAAO,IAAI,CAAC,SAAS,CAAC;IACxB,CAAC;IAED,IAAW,YAAY;QACrB,OAAO,IAAI,CAAC,SAAS,CAAC,QAAQ,EAAE,GAAG,CAAC,CAAC;IACvC,CAAC;IAED,IAAW,aAAa;QACtB,OAAO,IAAI,CAAC,SAAS,CAAC,QAAQ,EAAE,KAAK,CAAC,CAAC;IACzC,CAAC;IAED;;;;gFAI4E;IAErE,OAAO,CAAC,GAAqC;QAClD,IAAI,CAAC,GAAG,GAAG,GAAG,CAAC;QACf,IAAI,CAAC,WAAW,EAAE,CAAC;IACrB,CAAC;IAED;;;;gFAI4E;IAEpE,kBAAkB;QACxB,IAAI,SAAS,GAAG,IAAI,CAAC,SAAS,CAAC,QAAQ,EAAE,CAAC;QAC1C,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,EAAE,SAAS,CAAC,CAAC;IACnC,CAAC;IAEO,kBAAkB;QACxB,IAAI,SAAS,GAAG,IAAI,CAAC,SAAS,CAAC,QAAQ,EAAE,CAAC;QAC1C,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,EAAE,SAAS,CAAC,CAAC;IACnC,CAAC;IAEO,WAAW;QAEjB,IAAI,CAAC,QAAQ,EAAE,CAAC;QAEhB,IAAI,IAAI,CAAC,YAAY,EAAE;YACrB,IAAI,CAAC,MAAM,GAAG,IAAI,CAAC,GAAG,CAAC,UAAU,CAAC,IAAI,CAAC,KAAK,CAAC;iBAC1C,SAAS,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;SAC1B;IACH,CAAC;IAEO,QAAQ;QACd,IAAI,IAAI,CAAC,MAAM,EAAE;YACf,IAAI,CAAC,MAAM,CAAC,WAAW,EAAE,CAAC;YAC1B,IAAI,CAAC,MAAM,GAAG,IAAI,CAAC;SACpB;IACH,CAAC;CACF","sourcesContent":["\nimport {BehaviorSubject, Observable, Subject, Subscription} from 'rxjs';\nimport {KafentEvent} from '../kafent-event';\nimport {share} from 'rxjs/operators';\nimport {ReactiveEventSource} from '../../event-source/standard/reactive-event-source';\n\nexport class KafentTopicSse {\n\n  /***************************************************************************\n   *                                                                         *\n   * Fields                                                                  *\n   *                                                                         *\n   **************************************************************************/\n\n  /**\n   * The cold observable which proxies to the\n   * sse channel and tracks the open connections.\n   *\n   */\n  private readonly source: Observable<KafentEvent>;\n\n  /**\n   * The shared, ref counting stream clients subscribe to.\n   */\n  private readonly hot: Observable<KafentEvent>;\n\n  /**\n   * Intermediate channel transporting events from the sse stream to the\n   */\n  private readonly inter = new Subject<KafentEvent>();\n\n  private sse: ReactiveEventSource<KafentEvent>;\n\n  private sseSub: Subscription;\n\n  /**\n   * The open sse connections. Should only be 0 or 1 since multiple connections\n   * are shared by the hot stream.\n   */\n  private readonly connected = new BehaviorSubject(0);\n\n  /***************************************************************************\n   *                                                                         *\n   * Constructor                                                             *\n   *                                                                         *\n   **************************************************************************/\n\n  constructor(\n    private readonly topic: string\n  ) {\n    this.source = new Observable((observer) => {\n\n      const sub = this.inter.subscribe(observer);\n\n      if (this.noneConnected) {\n        this.incrementConnected();\n        this.renewSseSub();\n      }\n\n      // Teardown\n      return () => {\n        sub.unsubscribe();\n        this.decrementConnected();\n        if (this.noneConnected) {\n          this.unsubSse();\n        }\n      };\n    });\n    this.hot = this.source.pipe(share());\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Properties                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  /**\n   * Gets the events.\n   * Ensure to unsubscribe each subscription to avoid a resource leak.\n   */\n  public get events(): Observable<KafentEvent> {\n    return this.hot;\n  }\n\n  /**\n   * Emits the current count of connected clients.\n   * If this value is 0, it means this kafent-topic is currently unused.\n   */\n  public get connectedCount(): Observable<number> {\n    return this.connected;\n  }\n\n  public get anyConnected(): boolean {\n    return this.connected.getValue() > 0;\n  }\n\n  public get noneConnected(): boolean {\n    return this.connected.getValue() === 0;\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Public API                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  public connect(sse: ReactiveEventSource<KafentEvent>): void {\n    this.sse = sse;\n    this.renewSseSub();\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Private methods                                                         *\n   *                                                                         *\n   **************************************************************************/\n\n  private incrementConnected(): void {\n    let connected = this.connected.getValue();\n    this.connected.next(++connected);\n  }\n\n  private decrementConnected(): void {\n    let connected = this.connected.getValue();\n    this.connected.next(--connected);\n  }\n\n  private renewSseSub(): void {\n\n    this.unsubSse();\n\n    if (this.anyConnected) {\n      this.sseSub = this.sse.eventsJson(this.topic)\n        .subscribe(this.inter);\n    }\n  }\n\n  private unsubSse(): void {\n    if (this.sseSub) {\n      this.sseSub.unsubscribe();\n      this.sseSub = null;\n    }\n  }\n}\n"]}
|