Я использовал этот шаблон для создания службы ServerSentEvent: Angular и событий, отправленных сервером Вот код службы:
import { Injectable, NgZone } from "@angular/core";
import { Observable } from "rxjs";
@Injectable({
providedIn: "root"
})
export class SseService {
constructor(private _zone: NgZone) { }
getServerSentEvent(url: string): Observable<any> {
return Observable.create(observer => {
const eventSource = this.getEventSource(url);
eventSource.onmessage = event => {
this._zone.run(() => {
observer.next(event.data);
console.log(event); // Here I see every received event
});
};
eventSource.onerror = error => {
this._zone.run(() => {
observer.error(error);
console.log(error);
});
};
// return () => eventSource.close();
});
}
private getEventSource(url: string): EventSource {
return new EventSource(url);
}
}
Затем я вызываю этот поток ss-событий в основной компонент:
ngOnInit() { //This subscribe works just for the first event no more
this.sseService.getServerSentEvent("http://localhost:8080/ssevents")
.subscribe(
data => {
this.eventmessage = data;
},
error => this.error = error
);
// But direct next EventSource onmessage() call works here:
this.source = new EventSource("http://localhost:8080/ssevents");
this.source.onmessage = event => {
this.zone.run(() => {
this.eventmessage = event.data;
console.log(event);
});
};
this.source.onerror = error => {
this.zone.run(() => {
this.error = error.type;
console.log(error);
});
};
}
Почему этот внутренний наблюдаемый объект в sseservice не работает? Только первый. Поэтому мне нужно использовать второй вариант - но там я получаю постоянную ошибку EventTarget (каждую секунду или полсекунды) без причин для описания ошибки, кроме Type = error, EventPhase = 2. Но поступают события, которые генерируются сервером сразу после вставки в mongodb.