@elderbyte/ngx-starter 17.6.4 → 17.7.1
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/features/event-source/fetch/public_api.mjs +3 -0
- package/esm2022/lib/features/event-source/fetch/reactive-fetch-event-source.mjs +217 -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 +266 -3
- 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 +109 -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,3 @@
|
|
|
1
|
+
export * from './reactive-fetch-event-source';
|
|
2
|
+
export * from './reactive-fetch-event-source.service';
|
|
3
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHVibGljX2FwaS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL2VsZGVyYnl0ZS9uZ3gtc3RhcnRlci9zcmMvbGliL2ZlYXR1cmVzL2V2ZW50LXNvdXJjZS9mZXRjaC9wdWJsaWNfYXBpLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLGNBQWMsK0JBQStCLENBQUM7QUFDOUMsY0FBYyx1Q0FBdUMsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCAqIGZyb20gJy4vcmVhY3RpdmUtZmV0Y2gtZXZlbnQtc291cmNlJztcbmV4cG9ydCAqIGZyb20gJy4vcmVhY3RpdmUtZmV0Y2gtZXZlbnQtc291cmNlLnNlcnZpY2UnO1xuXG5cbiJdfQ==
|
|
@@ -0,0 +1,217 @@
|
|
|
1
|
+
import { EventStreamContentType, fetchEventSource } from '@microsoft/fetch-event-source';
|
|
2
|
+
import { LoggerFactory } from '@elderbyte/ts-logger';
|
|
3
|
+
import { BehaviorSubject, Subject } from 'rxjs';
|
|
4
|
+
import { map } from 'rxjs/operators';
|
|
5
|
+
export var ReactiveEventSourceState;
|
|
6
|
+
(function (ReactiveEventSourceState) {
|
|
7
|
+
ReactiveEventSourceState["IDLE"] = "IDLE";
|
|
8
|
+
ReactiveEventSourceState["CONNECTING"] = "CONNECTING";
|
|
9
|
+
ReactiveEventSourceState["OPEN"] = "OPEN";
|
|
10
|
+
ReactiveEventSourceState["CLOSED"] = "CLOSED";
|
|
11
|
+
ReactiveEventSourceState["ERROR"] = "ERROR";
|
|
12
|
+
})(ReactiveEventSourceState || (ReactiveEventSourceState = {}));
|
|
13
|
+
export class ReactiveSSeMessage {
|
|
14
|
+
constructor(id, event, data, retry) {
|
|
15
|
+
this.id = id;
|
|
16
|
+
this.event = event;
|
|
17
|
+
this.data = data;
|
|
18
|
+
this.retry = retry;
|
|
19
|
+
}
|
|
20
|
+
}
|
|
21
|
+
export class ReactiveFetchEventSource {
|
|
22
|
+
/***************************************************************************
|
|
23
|
+
* *
|
|
24
|
+
* Static Builder *
|
|
25
|
+
* *
|
|
26
|
+
**************************************************************************/
|
|
27
|
+
static staticRequest(requestInit) {
|
|
28
|
+
if (!requestInit) {
|
|
29
|
+
throw new Error('You must provide a event source url!');
|
|
30
|
+
}
|
|
31
|
+
return new ReactiveFetchEventSource(() => requestInit);
|
|
32
|
+
}
|
|
33
|
+
static dynamicRequest(requestInitProvider) {
|
|
34
|
+
return new ReactiveFetchEventSource(requestInitProvider);
|
|
35
|
+
}
|
|
36
|
+
/***************************************************************************
|
|
37
|
+
* *
|
|
38
|
+
* Constructor *
|
|
39
|
+
* *
|
|
40
|
+
**************************************************************************/
|
|
41
|
+
constructor(requestInitProvider) {
|
|
42
|
+
this.requestInitProvider = requestInitProvider;
|
|
43
|
+
/***************************************************************************
|
|
44
|
+
* *
|
|
45
|
+
* Fields *
|
|
46
|
+
* *
|
|
47
|
+
**************************************************************************/
|
|
48
|
+
this.log = LoggerFactory.getLogger(this.constructor.name);
|
|
49
|
+
this.abortController = new AbortController();
|
|
50
|
+
this.state$ = new BehaviorSubject(ReactiveEventSourceState.IDLE);
|
|
51
|
+
this.eventTopics = new Map();
|
|
52
|
+
if (!requestInitProvider) {
|
|
53
|
+
throw new Error('You must provide a event source url provider!');
|
|
54
|
+
}
|
|
55
|
+
this.open();
|
|
56
|
+
}
|
|
57
|
+
/***************************************************************************
|
|
58
|
+
* *
|
|
59
|
+
* Properties *
|
|
60
|
+
* *
|
|
61
|
+
**************************************************************************/
|
|
62
|
+
get closed() {
|
|
63
|
+
return this._closed;
|
|
64
|
+
}
|
|
65
|
+
/***************************************************************************
|
|
66
|
+
* *
|
|
67
|
+
* Public API *
|
|
68
|
+
* *
|
|
69
|
+
**************************************************************************/
|
|
70
|
+
/**
|
|
71
|
+
* Open the event source.
|
|
72
|
+
*
|
|
73
|
+
* Note: The connection is opened automatically upon object creation.
|
|
74
|
+
* This method should only be used if this reactive-event-source has
|
|
75
|
+
* been closed explicitly.
|
|
76
|
+
*/
|
|
77
|
+
open() {
|
|
78
|
+
this._closed = false;
|
|
79
|
+
this.reconnect();
|
|
80
|
+
}
|
|
81
|
+
/**
|
|
82
|
+
* Get an event stream for the given event type.
|
|
83
|
+
* @param eventTypeRaw The event type. Defaults to 'message'.
|
|
84
|
+
*/
|
|
85
|
+
events(eventTypeRaw) {
|
|
86
|
+
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();
|
|
92
|
+
}
|
|
93
|
+
/**
|
|
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
|
+
*/
|
|
99
|
+
eventsJson(eventType) {
|
|
100
|
+
return this.events(eventType).pipe(map(event => JSON.parse(event.data)));
|
|
101
|
+
}
|
|
102
|
+
/**
|
|
103
|
+
* Close this event source. It won't reconnect.
|
|
104
|
+
*/
|
|
105
|
+
close() {
|
|
106
|
+
this._closed = true;
|
|
107
|
+
this.closeCurrent();
|
|
108
|
+
this.log.debug('Closing the event-source.');
|
|
109
|
+
}
|
|
110
|
+
/**
|
|
111
|
+
* Keeps this reactive event source open until the given observable emits an event.
|
|
112
|
+
* @param unsubscribe
|
|
113
|
+
*/
|
|
114
|
+
openUntil(unsubscribe) {
|
|
115
|
+
this.cleanUpOpenUntil();
|
|
116
|
+
this._unsubscribeSub = unsubscribe
|
|
117
|
+
.subscribe(() => this.close());
|
|
118
|
+
return this;
|
|
119
|
+
}
|
|
120
|
+
/***************************************************************************
|
|
121
|
+
* *
|
|
122
|
+
* Private methods *
|
|
123
|
+
* *
|
|
124
|
+
**************************************************************************/
|
|
125
|
+
cleanUpOpenUntil() {
|
|
126
|
+
if (this._unsubscribeSub) {
|
|
127
|
+
this._unsubscribeSub.unsubscribe();
|
|
128
|
+
this._unsubscribeSub = null;
|
|
129
|
+
}
|
|
130
|
+
}
|
|
131
|
+
/**
|
|
132
|
+
* Close the current event source
|
|
133
|
+
*/
|
|
134
|
+
closeCurrent(reason) {
|
|
135
|
+
this.log.debug('Closing the event-source, reason: ' + reason);
|
|
136
|
+
this.abortController.abort(reason);
|
|
137
|
+
}
|
|
138
|
+
/**
|
|
139
|
+
* Reconnects this event source, unless closed is true;
|
|
140
|
+
*/
|
|
141
|
+
reconnect() {
|
|
142
|
+
if (this._closed) {
|
|
143
|
+
return;
|
|
144
|
+
}
|
|
145
|
+
this.updateState(ReactiveEventSourceState.CONNECTING);
|
|
146
|
+
this.closeCurrent('Reconnect');
|
|
147
|
+
this.openEventSource();
|
|
148
|
+
}
|
|
149
|
+
openEventSource() {
|
|
150
|
+
const requestInit = this.requestInitProvider();
|
|
151
|
+
fetchEventSource(requestInit.request, {
|
|
152
|
+
method: requestInit.method,
|
|
153
|
+
signal: this.abortController.signal,
|
|
154
|
+
headers: requestInit.headers,
|
|
155
|
+
keepalive: true,
|
|
156
|
+
openWhenHidden: false,
|
|
157
|
+
onopen: response => this.handleOnOpen(response),
|
|
158
|
+
onmessage: msg => this.handleOnMessage(msg),
|
|
159
|
+
onerror: err => this.handleOnError(err),
|
|
160
|
+
onclose: () => this.handleOnClose()
|
|
161
|
+
}).then(r => {
|
|
162
|
+
});
|
|
163
|
+
}
|
|
164
|
+
handleOnOpen(response) {
|
|
165
|
+
if (response.ok && response.headers.get('content-type') === EventStreamContentType) {
|
|
166
|
+
this.log.debug('EventSource connection sucessfully opened to: ' + response.url +
|
|
167
|
+
', state: ' + response.status, response);
|
|
168
|
+
this.updateState(ReactiveEventSourceState.OPEN);
|
|
169
|
+
return; // everything's good
|
|
170
|
+
}
|
|
171
|
+
else if (response.status >= 400 && response.status < 500 && response.status !== 429) {
|
|
172
|
+
// client-side errors are usually non-retriable:
|
|
173
|
+
this.updateState(ReactiveEventSourceState.ERROR);
|
|
174
|
+
this.log.error('Failed to open Event Source due to Client Error ' + response.status, response);
|
|
175
|
+
}
|
|
176
|
+
else {
|
|
177
|
+
this.log.error('Failed to open Event Source due to Error ' + response.status, response);
|
|
178
|
+
this.updateState(ReactiveEventSourceState.ERROR);
|
|
179
|
+
this.tryReconnect(); // TODO Necessary if keep alive = true?
|
|
180
|
+
}
|
|
181
|
+
}
|
|
182
|
+
handleOnError(err) {
|
|
183
|
+
this.updateState(ReactiveEventSourceState.ERROR);
|
|
184
|
+
this.log.trace('There was an SSE error.', err);
|
|
185
|
+
if (!this.closed) {
|
|
186
|
+
// There was an error - try reconnecting
|
|
187
|
+
this.log.debug('Attempting to reconnect event-source ...');
|
|
188
|
+
this.tryReconnect(); // TODO Necessary if keep alive = true?
|
|
189
|
+
}
|
|
190
|
+
else {
|
|
191
|
+
this.log.debug('There was an error in the sse connection (closed).', err);
|
|
192
|
+
}
|
|
193
|
+
}
|
|
194
|
+
tryReconnect() {
|
|
195
|
+
if (!this._closed) {
|
|
196
|
+
setTimeout(() => this.reconnect(), 3000); // Delay the reconnect
|
|
197
|
+
}
|
|
198
|
+
}
|
|
199
|
+
handleOnMessage(msg) {
|
|
200
|
+
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
|
+
}
|
|
205
|
+
}
|
|
206
|
+
handleOnClose() {
|
|
207
|
+
this.log.debug('EventSource has closed.');
|
|
208
|
+
this.updateState(ReactiveEventSourceState.CLOSED);
|
|
209
|
+
}
|
|
210
|
+
updateState(state) {
|
|
211
|
+
this.state$.next(state);
|
|
212
|
+
}
|
|
213
|
+
eventTypeOrDefault(type) {
|
|
214
|
+
return type ?? 'message';
|
|
215
|
+
}
|
|
216
|
+
}
|
|
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":"AAAA,OAAO,EAAqB,sBAAsB,EAAE,gBAAgB,EAAC,MAAM,+BAA+B,CAAC;AAC3G,OAAO,EAAC,aAAa,EAAC,MAAM,sBAAsB,CAAC;AACnD,OAAO,EAAC,eAAe,EAAc,OAAO,EAAe,MAAM,MAAM,CAAC;AACxE,OAAO,EAAC,GAAG,EAAC,MAAM,gBAAgB,CAAC;AAEnC,MAAM,CAAN,IAAY,wBAMX;AAND,WAAY,wBAAwB;IAClC,yCAAa,CAAA;IACb,qDAAyB,CAAA;IACzB,yCAAa,CAAA;IACb,6CAAiB,CAAA;IACjB,2CAAe,CAAA;AACjB,CAAC,EANW,wBAAwB,KAAxB,wBAAwB,QAMnC;AAED,MAAM,OAAO,kBAAkB;IAC7B,YACkB,EAAU,EACV,KAAa,EACb,IAAO,EACP,KAAc;QAHd,OAAE,GAAF,EAAE,CAAQ;QACV,UAAK,GAAL,KAAK,CAAQ;QACb,SAAI,GAAJ,IAAI,CAAG;QACP,UAAK,GAAL,KAAK,CAAS;IAC5B,CAAC;CACN;AAQD,MAAM,OAAO,wBAAwB;IAkBnC;;;;gFAI4E;IAErE,MAAM,CAAC,aAAa,CACzB,WAAmC;QAEnC,IAAI,CAAC,WAAW,EAAE;YAChB,MAAM,IAAI,KAAK,CAAC,sCAAsC,CAAC,CAAC;SACzD;QACD,OAAO,IAAI,wBAAwB,CAAC,GAAG,EAAE,CAAC,WAAW,CAAC,CAAC;IACzD,CAAC;IAEM,MAAM,CAAC,cAAc,CAC1B,mBAAiD;QAEjD,OAAO,IAAI,wBAAwB,CAAC,mBAAmB,CAAC,CAAC;IAC3D,CAAC;IAED;;;;gFAI4E;IAE5E,YACmB,mBAAiD;QAAjD,wBAAmB,GAAnB,mBAAmB,CAA8B;QA5CpE;;;;oFAI4E;QAE3D,QAAG,GAAG,aAAa,CAAC,SAAS,CAAC,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,CAAC;QAErD,oBAAe,GAAG,IAAI,eAAe,EAAE,CAAC;QAExC,WAAM,GAAG,IAAI,eAAe,CAA2B,wBAAwB,CAAC,IAAI,CAAC,CAAC;QAGtF,gBAAW,GAAG,IAAI,GAAG,EAAuC,CAAC;QAiC5E,IAAI,CAAC,mBAAmB,EAAE;YACxB,MAAM,IAAI,KAAK,CAAC,+CAA+C,CAAC,CAAC;SAClE;QACD,IAAI,CAAC,IAAI,EAAE,CAAC;IACd,CAAC;IAED;;;;gFAI4E;IAE5E,IAAW,MAAM;QACf,OAAO,IAAI,CAAC,OAAO,CAAC;IACtB,CAAC;IAED;;;;gFAI4E;IAE5E;;;;;;OAMG;IACI,IAAI;QACT,IAAI,CAAC,OAAO,GAAG,KAAK,CAAC;QACrB,IAAI,CAAC,SAAS,EAAE,CAAC;IACnB,CAAC;IAED;;;OAGG;IACI,MAAM,CAAC,YAAqB;QACjC,MAAM,SAAS,GAAG,IAAI,CAAC,kBAAkB,CAAC,YAAY,CAAC,CAAC;QACxD,IAAI,CAAC,IAAI,CAAC,WAAW,CAAC,GAAG,CAAC,SAAS,CAAC,EAAE;YACpC,IAAI,CAAC,WAAW,CAAC,GAAG,CAAC,SAAS,EAAE,IAAI,OAAO,EAAsB,CAAC,CAAC;SACpE;QACD,MAAM,SAAS,GAAG,IAAI,CAAC,WAAW,CAAC,GAAG,CAAC,SAAS,CAAC,CAAC;QAClD,OAAO,SAAS,CAAC,YAAY,EAAE,CAAC;IAClC,CAAC;IAED;;;;;OAKG;IACI,UAAU,CAAC,SAAkB;QAClC,OAAO,IAAI,CAAC,MAAM,CAAC,SAAS,CAAC,CAAC,IAAI,CAChC,GAAG,CAAC,KAAK,CAAC,EAAE,CAAC,IAAI,CAAC,KAAK,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC,CACrC,CAAC;IACJ,CAAC;IAED;;OAEG;IACI,KAAK;QACV,IAAI,CAAC,OAAO,GAAG,IAAI,CAAC;QACpB,IAAI,CAAC,YAAY,EAAE,CAAC;QACpB,IAAI,CAAC,GAAG,CAAC,KAAK,CAAC,2BAA2B,CAAC,CAAC;IAC9C,CAAC;IAED;;;OAGG;IACI,SAAS,CAAC,WAA4B;QAC3C,IAAI,CAAC,gBAAgB,EAAE,CAAC;QACxB,IAAI,CAAC,eAAe,GAAG,WAAW;aAC/B,SAAS,CACR,GAAG,EAAE,CAAC,IAAI,CAAC,KAAK,EAAE,CACnB,CAAC;QACJ,OAAO,IAAI,CAAC;IACd,CAAC;IAED;;;;gFAI4E;IAEpE,gBAAgB;QACtB,IAAI,IAAI,CAAC,eAAe,EAAE;YACxB,IAAI,CAAC,eAAe,CAAC,WAAW,EAAE,CAAC;YACnC,IAAI,CAAC,eAAe,GAAG,IAAI,CAAC;SAC7B;IACH,CAAC;IAED;;OAEG;IACK,YAAY,CAAC,MAAe;QAClC,IAAI,CAAC,GAAG,CAAC,KAAK,CAAC,oCAAoC,GAAG,MAAM,CAAC,CAAC;QAC9D,IAAI,CAAC,eAAe,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC;IACrC,CAAC;IAED;;OAEG;IACK,SAAS;QACf,IAAI,IAAI,CAAC,OAAO,EAAE;YAChB,OAAO;SACR;QACD,IAAI,CAAC,WAAW,CAAC,wBAAwB,CAAC,UAAU,CAAC,CAAC;QACtD,IAAI,CAAC,YAAY,CAAC,WAAW,CAAC,CAAC;QAC/B,IAAI,CAAC,eAAe,EAAE,CAAC;IACzB,CAAC;IAEO,eAAe;QACrB,MAAM,WAAW,GAAG,IAAI,CAAC,mBAAmB,EAAE,CAAC;QAC/C,gBAAgB,CAAC,WAAW,CAAC,OAAO,EAAE;YACpC,MAAM,EAAE,WAAW,CAAC,MAAM;YAC1B,MAAM,EAAE,IAAI,CAAC,eAAe,CAAC,MAAM;YACnC,OAAO,EAAE,WAAW,CAAC,OAAO;YAC5B,SAAS,EAAE,IAAI;YACf,cAAc,EAAE,KAAK;YACrB,MAAM,EAAE,QAAQ,CAAC,EAAE,CAAC,IAAI,CAAC,YAAY,CAAC,QAAQ,CAAC;YAC/C,SAAS,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,CAAC,eAAe,CAAC,GAAG,CAAC;YAC3C,OAAO,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,CAAC,aAAa,CAAC,GAAG,CAAC;YACvC,OAAO,EAAE,GAAG,EAAE,CAAC,IAAI,CAAC,aAAa,EAAE;SACpC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE;QAEZ,CAAC,CAAC,CAAA;IACJ,CAAC;IAEO,YAAY,CAAC,QAAkB;QACrC,IAAI,QAAQ,CAAC,EAAE,IAAI,QAAQ,CAAC,OAAO,CAAC,GAAG,CAAC,cAAc,CAAC,KAAK,sBAAsB,EAAE;YAClF,IAAI,CAAC,GAAG,CAAC,KAAK,CAAC,gDAAgD,GAAG,QAAQ,CAAC,GAAG;gBAC5E,WAAW,GAAG,QAAQ,CAAC,MAAM,EAAE,QAAQ,CAAC,CAAC;YAC3C,IAAI,CAAC,WAAW,CAAC,wBAAwB,CAAC,IAAI,CAAC,CAAC;YAChD,OAAO,CAAC,oBAAoB;SAC7B;aAAM,IAAI,QAAQ,CAAC,MAAM,IAAI,GAAG,IAAI,QAAQ,CAAC,MAAM,GAAG,GAAG,IAAI,QAAQ,CAAC,MAAM,KAAK,GAAG,EAAE;YACrF,gDAAgD;YAChD,IAAI,CAAC,WAAW,CAAC,wBAAwB,CAAC,KAAK,CAAC,CAAC;YACjD,IAAI,CAAC,GAAG,CAAC,KAAK,CAAC,kDAAkD,GAAG,QAAQ,CAAC,MAAM,EAAE,QAAQ,CAAC,CAAC;SAChG;aAAM;YACL,IAAI,CAAC,GAAG,CAAC,KAAK,CAAC,2CAA2C,GAAG,QAAQ,CAAC,MAAM,EAAE,QAAQ,CAAC,CAAC;YACxF,IAAI,CAAC,WAAW,CAAC,wBAAwB,CAAC,KAAK,CAAC,CAAC;YACjD,IAAI,CAAC,YAAY,EAAE,CAAC,CAAC,uCAAuC;SAC7D;IACH,CAAC;IAEO,aAAa,CAAC,GAAQ;QAC5B,IAAI,CAAC,WAAW,CAAC,wBAAwB,CAAC,KAAK,CAAC,CAAC;QACjD,IAAI,CAAC,GAAG,CAAC,KAAK,CAAC,yBAAyB,EAAE,GAAG,CAAC,CAAC;QAC/C,IAAI,CAAC,IAAI,CAAC,MAAM,EAAE;YAChB,wCAAwC;YACxC,IAAI,CAAC,GAAG,CAAC,KAAK,CAAC,0CAA0C,CAAC,CAAC;YAC3D,IAAI,CAAC,YAAY,EAAE,CAAC,CAAC,uCAAuC;SAC7D;aAAM;YACL,IAAI,CAAC,GAAG,CAAC,KAAK,CAAC,oDAAoD,EAAE,GAAG,CAAC,CAAC;SAC3E;IACH,CAAC;IAEO,YAAY;QAClB,IAAI,CAAC,IAAI,CAAC,OAAO,EAAE;YACjB,UAAU,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,SAAS,EAAE,EAAE,IAAI,CAAC,CAAC,CAAC,sBAAsB;SACjE;IACH,CAAC;IAEO,eAAe,CAAC,GAAuB;QAC7C,MAAM,SAAS,GAAG,IAAI,CAAC,kBAAkB,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC;QACrD,MAAM,KAAK,GAAG,IAAI,CAAC,WAAW,CAAC,GAAG,CAAC,SAAS,CAAC,CAAC;QAC9C,IAAI,KAAK,EAAE;YACT,KAAK,CAAC,IAAI,CAAC,IAAI,kBAAkB,CAC/B,GAAG,CAAC,EAAE,EACN,SAAS,EACT,GAAG,CAAC,IAAI,EACR,GAAG,CAAC,KAAK,CACV,CAAC,CAAA;SACH;IACH,CAAC;IAEO,aAAa;QACnB,IAAI,CAAC,GAAG,CAAC,KAAK,CAAC,yBAAyB,CAAC,CAAA;QACzC,IAAI,CAAC,WAAW,CAAC,wBAAwB,CAAC,MAAM,CAAC,CAAC;IACpD,CAAC;IAEO,WAAW,CAAC,KAA+B;QACjD,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;IAC1B,CAAC;IAEO,kBAAkB,CAAC,IAAa;QACtC,OAAO,IAAI,IAAI,SAAS,CAAC;IAC3B,CAAC;CACF","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"]}
|
|
@@ -0,0 +1,53 @@
|
|
|
1
|
+
import { Injectable } from '@angular/core';
|
|
2
|
+
import { LoggerFactory } from '@elderbyte/ts-logger';
|
|
3
|
+
import { ReactiveFetchEventSource } from './reactive-fetch-event-source';
|
|
4
|
+
import * as i0 from "@angular/core";
|
|
5
|
+
/**
|
|
6
|
+
* Angular reactive EventSource integration
|
|
7
|
+
*/
|
|
8
|
+
export class ReactiveFetchEventSourceService {
|
|
9
|
+
/***************************************************************************
|
|
10
|
+
* *
|
|
11
|
+
* Constructor *
|
|
12
|
+
* *
|
|
13
|
+
**************************************************************************/
|
|
14
|
+
constructor() {
|
|
15
|
+
/***************************************************************************
|
|
16
|
+
* *
|
|
17
|
+
* Fields *
|
|
18
|
+
* *
|
|
19
|
+
**************************************************************************/
|
|
20
|
+
this.log = LoggerFactory.getLogger(this.constructor.name);
|
|
21
|
+
}
|
|
22
|
+
/***************************************************************************
|
|
23
|
+
* *
|
|
24
|
+
* Public API *
|
|
25
|
+
* *
|
|
26
|
+
**************************************************************************/
|
|
27
|
+
/**
|
|
28
|
+
* Creates an reactive, automatic reconnecting event source.
|
|
29
|
+
*
|
|
30
|
+
* @param requestInit The request to the event source
|
|
31
|
+
*/
|
|
32
|
+
reactiveEventSource(requestInit) {
|
|
33
|
+
return ReactiveFetchEventSource.staticRequest(requestInit);
|
|
34
|
+
}
|
|
35
|
+
/**
|
|
36
|
+
* Creates an reactive, automatic reconnecting event source. Each time the event source
|
|
37
|
+
* reconnects, the url provider is invoked.
|
|
38
|
+
*
|
|
39
|
+
* @param requestInitProvider The request provider of the event source
|
|
40
|
+
*/
|
|
41
|
+
reactiveEventSourceDynamic(requestInitProvider) {
|
|
42
|
+
return ReactiveFetchEventSource.dynamicRequest(requestInitProvider);
|
|
43
|
+
}
|
|
44
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.4", ngImport: i0, type: ReactiveFetchEventSourceService, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); }
|
|
45
|
+
static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "17.3.4", ngImport: i0, type: ReactiveFetchEventSourceService, providedIn: 'root' }); }
|
|
46
|
+
}
|
|
47
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.4", ngImport: i0, type: ReactiveFetchEventSourceService, decorators: [{
|
|
48
|
+
type: Injectable,
|
|
49
|
+
args: [{
|
|
50
|
+
providedIn: 'root'
|
|
51
|
+
}]
|
|
52
|
+
}], ctorParameters: () => [] });
|
|
53
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicmVhY3RpdmUtZmV0Y2gtZXZlbnQtc291cmNlLnNlcnZpY2UuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9lbGRlcmJ5dGUvbmd4LXN0YXJ0ZXIvc3JjL2xpYi9mZWF0dXJlcy9ldmVudC1zb3VyY2UvZmV0Y2gvcmVhY3RpdmUtZmV0Y2gtZXZlbnQtc291cmNlLnNlcnZpY2UudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFDLFVBQVUsRUFBUyxNQUFNLGVBQWUsQ0FBQztBQUNqRCxPQUFPLEVBQUMsYUFBYSxFQUFDLE1BQU0sc0JBQXNCLENBQUM7QUFDbkQsT0FBTyxFQUF5Qix3QkFBd0IsRUFBQyxNQUFNLCtCQUErQixDQUFDOztBQUUvRjs7R0FFRztBQUlILE1BQU0sT0FBTywrQkFBK0I7SUFVMUM7Ozs7Z0ZBSTRFO0lBRTVFO1FBZEE7Ozs7b0ZBSTRFO1FBRTNELFFBQUcsR0FBRyxhQUFhLENBQUMsU0FBUyxDQUFDLElBQUksQ0FBQyxXQUFXLENBQUMsSUFBSSxDQUFDLENBQUM7SUFRdEQsQ0FBQztJQUVqQjs7OztnRkFJNEU7SUFFNUU7Ozs7T0FJRztJQUNJLG1CQUFtQixDQUN4QixXQUFtQztRQUVuQyxPQUFPLHdCQUF3QixDQUFDLGFBQWEsQ0FBQyxXQUFXLENBQUMsQ0FBQztJQUM3RCxDQUFDO0lBRUQ7Ozs7O09BS0c7SUFDSSwwQkFBMEIsQ0FDL0IsbUJBQWlEO1FBRWpELE9BQU8sd0JBQXdCLENBQUMsY0FBYyxDQUM1QyxtQkFBbUIsQ0FDcEIsQ0FBQztJQUNKLENBQUM7OEdBL0NVLCtCQUErQjtrSEFBL0IsK0JBQStCLGNBRjlCLE1BQU07OzJGQUVQLCtCQUErQjtrQkFIM0MsVUFBVTttQkFBQztvQkFDVixVQUFVLEVBQUUsTUFBTTtpQkFDbkIiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQge0luamVjdGFibGUsIE5nWm9uZX0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQge0xvZ2dlckZhY3Rvcnl9IGZyb20gJ0BlbGRlcmJ5dGUvdHMtbG9nZ2VyJztcbmltcG9ydCB7RXZlbnRTb3VyY2VSZXF1ZXN0SW5pdCwgUmVhY3RpdmVGZXRjaEV2ZW50U291cmNlfSBmcm9tICcuL3JlYWN0aXZlLWZldGNoLWV2ZW50LXNvdXJjZSc7XG5cbi8qKlxuICogQW5ndWxhciByZWFjdGl2ZSBFdmVudFNvdXJjZSBpbnRlZ3JhdGlvblxuICovXG5ASW5qZWN0YWJsZSh7XG4gIHByb3ZpZGVkSW46ICdyb290J1xufSlcbmV4cG9ydCBjbGFzcyBSZWFjdGl2ZUZldGNoRXZlbnRTb3VyY2VTZXJ2aWNlIHtcblxuICAvKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqXG4gICAqICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICpcbiAgICogRmllbGRzICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgKlxuICAgKiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAqXG4gICAqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKi9cblxuICBwcml2YXRlIHJlYWRvbmx5IGxvZyA9IExvZ2dlckZhY3RvcnkuZ2V0TG9nZ2VyKHRoaXMuY29uc3RydWN0b3IubmFtZSk7XG5cbiAgLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKlxuICAgKiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAqXG4gICAqIENvbnN0cnVjdG9yICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICpcbiAgICogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgKlxuICAgKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKiovXG5cbiAgY29uc3RydWN0b3IoKSB7IH1cblxuICAvKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqXG4gICAqICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICpcbiAgICogUHVibGljIEFQSSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgKlxuICAgKiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAqXG4gICAqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKi9cblxuICAvKipcbiAgICogQ3JlYXRlcyBhbiByZWFjdGl2ZSwgYXV0b21hdGljIHJlY29ubmVjdGluZyBldmVudCBzb3VyY2UuXG4gICAqXG4gICAqIEBwYXJhbSByZXF1ZXN0SW5pdCBUaGUgcmVxdWVzdCB0byB0aGUgZXZlbnQgc291cmNlXG4gICAqL1xuICBwdWJsaWMgcmVhY3RpdmVFdmVudFNvdXJjZTxUID0gYW55PihcbiAgICByZXF1ZXN0SW5pdDogRXZlbnRTb3VyY2VSZXF1ZXN0SW5pdFxuICApOiBSZWFjdGl2ZUZldGNoRXZlbnRTb3VyY2U8VD4ge1xuICAgIHJldHVybiBSZWFjdGl2ZUZldGNoRXZlbnRTb3VyY2Uuc3RhdGljUmVxdWVzdChyZXF1ZXN0SW5pdCk7XG4gIH1cblxuICAvKipcbiAgICogQ3JlYXRlcyBhbiByZWFjdGl2ZSwgYXV0b21hdGljIHJlY29ubmVjdGluZyBldmVudCBzb3VyY2UuIEVhY2ggdGltZSB0aGUgZXZlbnQgc291cmNlXG4gICAqIHJlY29ubmVjdHMsIHRoZSB1cmwgcHJvdmlkZXIgaXMgaW52b2tlZC5cbiAgICpcbiAgICogQHBhcmFtIHJlcXVlc3RJbml0UHJvdmlkZXIgVGhlIHJlcXVlc3QgcHJvdmlkZXIgb2YgdGhlIGV2ZW50IHNvdXJjZVxuICAgKi9cbiAgcHVibGljIHJlYWN0aXZlRXZlbnRTb3VyY2VEeW5hbWljPFQgPSBhbnk+KFxuICAgIHJlcXVlc3RJbml0UHJvdmlkZXI6ICgpID0+IEV2ZW50U291cmNlUmVxdWVzdEluaXRcbiAgKTogUmVhY3RpdmVGZXRjaEV2ZW50U291cmNlPFQ+IHtcbiAgICByZXR1cm4gUmVhY3RpdmVGZXRjaEV2ZW50U291cmNlLmR5bmFtaWNSZXF1ZXN0KFxuICAgICAgcmVxdWVzdEluaXRQcm92aWRlclxuICAgICk7XG4gIH1cblxufVxuIl19
|
|
@@ -1,3 +1,3 @@
|
|
|
1
|
-
export * from './
|
|
2
|
-
export * from './
|
|
3
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
1
|
+
export * from './standard/public_api';
|
|
2
|
+
export * from './fetch/public_api';
|
|
3
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHVibGljX2FwaS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL2VsZGVyYnl0ZS9uZ3gtc3RhcnRlci9zcmMvbGliL2ZlYXR1cmVzL2V2ZW50LXNvdXJjZS9wdWJsaWNfYXBpLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLGNBQWMsdUJBQXVCLENBQUM7QUFDdEMsY0FBYyxvQkFBb0IsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCAqIGZyb20gJy4vc3RhbmRhcmQvcHVibGljX2FwaSc7XG5leHBvcnQgKiBmcm9tICcuL2ZldGNoL3B1YmxpY19hcGknO1xuXG4iXX0=
|
|
@@ -0,0 +1,56 @@
|
|
|
1
|
+
import { Injectable } from '@angular/core';
|
|
2
|
+
import { LoggerFactory } from '@elderbyte/ts-logger';
|
|
3
|
+
import { ReactiveEventSource } from './reactive-event-source';
|
|
4
|
+
import * as i0 from "@angular/core";
|
|
5
|
+
/**
|
|
6
|
+
* Angular reactive EventSource integration
|
|
7
|
+
*/
|
|
8
|
+
export class ElderEventSourceService {
|
|
9
|
+
/***************************************************************************
|
|
10
|
+
* *
|
|
11
|
+
* Constructor *
|
|
12
|
+
* *
|
|
13
|
+
**************************************************************************/
|
|
14
|
+
constructor(zone) {
|
|
15
|
+
this.zone = zone;
|
|
16
|
+
/***************************************************************************
|
|
17
|
+
* *
|
|
18
|
+
* Fields *
|
|
19
|
+
* *
|
|
20
|
+
**************************************************************************/
|
|
21
|
+
this.logger = LoggerFactory.getLogger(this.constructor.name);
|
|
22
|
+
}
|
|
23
|
+
/***************************************************************************
|
|
24
|
+
* *
|
|
25
|
+
* Public API *
|
|
26
|
+
* *
|
|
27
|
+
**************************************************************************/
|
|
28
|
+
/**
|
|
29
|
+
* Creates an reactive, automatic reconnecting event source.
|
|
30
|
+
*
|
|
31
|
+
* @param eventSourceUrl The url to the event source
|
|
32
|
+
* @param eventSourceInitDict Additional configuration to use when connecting. (optional)
|
|
33
|
+
*/
|
|
34
|
+
reactiveEventSource(eventSourceUrl, eventSourceInitDict) {
|
|
35
|
+
return ReactiveEventSource.staticUrl(this.zone, eventSourceUrl, eventSourceInitDict);
|
|
36
|
+
}
|
|
37
|
+
/**
|
|
38
|
+
* Creates an reactive, automatic reconnecting event source. Each time the event source
|
|
39
|
+
* reconnects, the url provider is invoked.
|
|
40
|
+
*
|
|
41
|
+
* @param eventSourceUrlProvider The url provider of the event source
|
|
42
|
+
* @param eventSourceInitDict Additional configuration to use when connecting. (optional)
|
|
43
|
+
*/
|
|
44
|
+
reactiveEventSourceDynamic(eventSourceUrlProvider, eventSourceInitDict) {
|
|
45
|
+
return ReactiveEventSource.dynamicUrl(this.zone, eventSourceUrlProvider, eventSourceInitDict);
|
|
46
|
+
}
|
|
47
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.4", ngImport: i0, type: ElderEventSourceService, deps: [{ token: i0.NgZone }], target: i0.ɵɵFactoryTarget.Injectable }); }
|
|
48
|
+
static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "17.3.4", ngImport: i0, type: ElderEventSourceService, providedIn: 'root' }); }
|
|
49
|
+
}
|
|
50
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.4", ngImport: i0, type: ElderEventSourceService, decorators: [{
|
|
51
|
+
type: Injectable,
|
|
52
|
+
args: [{
|
|
53
|
+
providedIn: 'root'
|
|
54
|
+
}]
|
|
55
|
+
}], ctorParameters: () => [{ type: i0.NgZone }] });
|
|
56
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZWxkZXItZXZlbnQtc291cmNlLnNlcnZpY2UuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9lbGRlcmJ5dGUvbmd4LXN0YXJ0ZXIvc3JjL2xpYi9mZWF0dXJlcy9ldmVudC1zb3VyY2Uvc3RhbmRhcmQvZWxkZXItZXZlbnQtc291cmNlLnNlcnZpY2UudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFDLFVBQVUsRUFBUyxNQUFNLGVBQWUsQ0FBQztBQUNqRCxPQUFPLEVBQUMsYUFBYSxFQUFDLE1BQU0sc0JBQXNCLENBQUM7QUFDbkQsT0FBTyxFQUFDLG1CQUFtQixFQUFDLE1BQU0seUJBQXlCLENBQUM7O0FBRTVEOztHQUVHO0FBSUgsTUFBTSxPQUFPLHVCQUF1QjtJQVVsQzs7OztnRkFJNEU7SUFFNUUsWUFDVSxJQUFZO1FBQVosU0FBSSxHQUFKLElBQUksQ0FBUTtRQWZ0Qjs7OztvRkFJNEU7UUFFM0QsV0FBTSxHQUFHLGFBQWEsQ0FBQyxTQUFTLENBQUMsSUFBSSxDQUFDLFdBQVcsQ0FBQyxJQUFJLENBQUMsQ0FBQztJQVVyRSxDQUFDO0lBRUw7Ozs7Z0ZBSTRFO0lBRTVFOzs7OztPQUtHO0lBQ0ksbUJBQW1CLENBQVUsY0FBc0IsRUFDdEIsbUJBQXFDO1FBRXZFLE9BQU8sbUJBQW1CLENBQUMsU0FBUyxDQUNsQyxJQUFJLENBQUMsSUFBSSxFQUNULGNBQWMsRUFDZCxtQkFBbUIsQ0FDcEIsQ0FBQztJQUNKLENBQUM7SUFFRDs7Ozs7O09BTUc7SUFDSSwwQkFBMEIsQ0FDL0Isc0JBQW9DLEVBQ3BDLG1CQUFxQztRQUVyQyxPQUFPLG1CQUFtQixDQUFDLFVBQVUsQ0FDbkMsSUFBSSxDQUFDLElBQUksRUFDVCxzQkFBc0IsRUFDdEIsbUJBQW1CLENBQ3BCLENBQUM7SUFDSixDQUFDOzhHQTFEVSx1QkFBdUI7a0hBQXZCLHVCQUF1QixjQUZ0QixNQUFNOzsyRkFFUCx1QkFBdUI7a0JBSG5DLFVBQVU7bUJBQUM7b0JBQ1YsVUFBVSxFQUFFLE1BQU07aUJBQ25CIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHtJbmplY3RhYmxlLCBOZ1pvbmV9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHtMb2dnZXJGYWN0b3J5fSBmcm9tICdAZWxkZXJieXRlL3RzLWxvZ2dlcic7XG5pbXBvcnQge1JlYWN0aXZlRXZlbnRTb3VyY2V9IGZyb20gJy4vcmVhY3RpdmUtZXZlbnQtc291cmNlJztcblxuLyoqXG4gKiBBbmd1bGFyIHJlYWN0aXZlIEV2ZW50U291cmNlIGludGVncmF0aW9uXG4gKi9cbkBJbmplY3RhYmxlKHtcbiAgcHJvdmlkZWRJbjogJ3Jvb3QnXG59KVxuZXhwb3J0IGNsYXNzIEVsZGVyRXZlbnRTb3VyY2VTZXJ2aWNlIHtcblxuICAvKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqXG4gICAqICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICpcbiAgICogRmllbGRzICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgKlxuICAgKiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAqXG4gICAqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKi9cblxuICBwcml2YXRlIHJlYWRvbmx5IGxvZ2dlciA9IExvZ2dlckZhY3RvcnkuZ2V0TG9nZ2VyKHRoaXMuY29uc3RydWN0b3IubmFtZSk7XG5cbiAgLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKlxuICAgKiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAqXG4gICAqIENvbnN0cnVjdG9yICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICpcbiAgICogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgKlxuICAgKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKiovXG5cbiAgY29uc3RydWN0b3IoXG4gICAgcHJpdmF0ZSB6b25lOiBOZ1pvbmVcbiAgKSB7IH1cblxuICAvKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqXG4gICAqICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICpcbiAgICogUHVibGljIEFQSSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgKlxuICAgKiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAqXG4gICAqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKi9cblxuICAvKipcbiAgICogQ3JlYXRlcyBhbiByZWFjdGl2ZSwgYXV0b21hdGljIHJlY29ubmVjdGluZyBldmVudCBzb3VyY2UuXG4gICAqXG4gICAqIEBwYXJhbSBldmVudFNvdXJjZVVybCBUaGUgdXJsIHRvIHRoZSBldmVudCBzb3VyY2VcbiAgICogQHBhcmFtIGV2ZW50U291cmNlSW5pdERpY3QgQWRkaXRpb25hbCBjb25maWd1cmF0aW9uIHRvIHVzZSB3aGVuIGNvbm5lY3RpbmcuIChvcHRpb25hbClcbiAgICovXG4gIHB1YmxpYyByZWFjdGl2ZUV2ZW50U291cmNlPFQgPSBhbnk+KGV2ZW50U291cmNlVXJsOiBzdHJpbmcsXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGV2ZW50U291cmNlSW5pdERpY3Q/OiBFdmVudFNvdXJjZUluaXRcbiAgKTogUmVhY3RpdmVFdmVudFNvdXJjZTxUPiB7XG4gICAgcmV0dXJuIFJlYWN0aXZlRXZlbnRTb3VyY2Uuc3RhdGljVXJsKFxuICAgICAgdGhpcy56b25lLFxuICAgICAgZXZlbnRTb3VyY2VVcmwsXG4gICAgICBldmVudFNvdXJjZUluaXREaWN0LFxuICAgICk7XG4gIH1cblxuICAvKipcbiAgICogQ3JlYXRlcyBhbiByZWFjdGl2ZSwgYXV0b21hdGljIHJlY29ubmVjdGluZyBldmVudCBzb3VyY2UuIEVhY2ggdGltZSB0aGUgZXZlbnQgc291cmNlXG4gICAqIHJlY29ubmVjdHMsIHRoZSB1cmwgcHJvdmlkZXIgaXMgaW52b2tlZC5cbiAgICpcbiAgICogQHBhcmFtIGV2ZW50U291cmNlVXJsUHJvdmlkZXIgVGhlIHVybCBwcm92aWRlciBvZiB0aGUgZXZlbnQgc291cmNlXG4gICAqIEBwYXJhbSBldmVudFNvdXJjZUluaXREaWN0IEFkZGl0aW9uYWwgY29uZmlndXJhdGlvbiB0byB1c2Ugd2hlbiBjb25uZWN0aW5nLiAob3B0aW9uYWwpXG4gICAqL1xuICBwdWJsaWMgcmVhY3RpdmVFdmVudFNvdXJjZUR5bmFtaWM8VCA9IGFueT4oXG4gICAgZXZlbnRTb3VyY2VVcmxQcm92aWRlcjogKCkgPT4gc3RyaW5nLFxuICAgIGV2ZW50U291cmNlSW5pdERpY3Q/OiBFdmVudFNvdXJjZUluaXRcbiAgKTogUmVhY3RpdmVFdmVudFNvdXJjZTxUPiB7XG4gICAgcmV0dXJuIFJlYWN0aXZlRXZlbnRTb3VyY2UuZHluYW1pY1VybChcbiAgICAgIHRoaXMuem9uZSxcbiAgICAgIGV2ZW50U291cmNlVXJsUHJvdmlkZXIsXG4gICAgICBldmVudFNvdXJjZUluaXREaWN0LFxuICAgICk7XG4gIH1cblxufVxuIl19
|
|
@@ -0,0 +1,3 @@
|
|
|
1
|
+
export * from './reactive-event-source';
|
|
2
|
+
export * from './elder-event-source.service';
|
|
3
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHVibGljX2FwaS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL2VsZGVyYnl0ZS9uZ3gtc3RhcnRlci9zcmMvbGliL2ZlYXR1cmVzL2V2ZW50LXNvdXJjZS9zdGFuZGFyZC9wdWJsaWNfYXBpLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLGNBQWMseUJBQXlCLENBQUM7QUFDeEMsY0FBYyw4QkFBOEIsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCAqIGZyb20gJy4vcmVhY3RpdmUtZXZlbnQtc291cmNlJztcbmV4cG9ydCAqIGZyb20gJy4vZWxkZXItZXZlbnQtc291cmNlLnNlcnZpY2UnO1xuXG4iXX0=
|
|
@@ -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"]}
|