aboutsummaryrefslogtreecommitdiff
path: root/src/js/builtins/EventStream.ts
diff options
context:
space:
mode:
Diffstat (limited to 'src/js/builtins/EventStream.ts')
-rw-r--r--src/js/builtins/EventStream.ts93
1 files changed, 80 insertions, 13 deletions
diff --git a/src/js/builtins/EventStream.ts b/src/js/builtins/EventStream.ts
index c9d574836..f55271228 100644
--- a/src/js/builtins/EventStream.ts
+++ b/src/js/builtins/EventStream.ts
@@ -1,29 +1,62 @@
+// https://html.spec.whatwg.org/multipage/server-sent-events.html
import type { EventStreamOptions, EventStream as IEventStream } from "bun";
export function getEventStream() {
- return class EventStream extends ReadableStream implements IEventStream {
- #ctrl: ReadableStreamDirectController | undefined;
+ class EventStream extends ReadableStream implements IEventStream {
+ // internal reference to the direct controller.
+ // we initialize it to a stub that writes to an internal queue.
+ // this makes it so you can call send() before the stream is started.
+ #ctrl: ReadableStreamDirectController | null;
+ // This field is read by `new Response`
+ $contentType = "text/event-stream";
- constructor(opts?: EventStreamOptions) {
+ constructor(opts: EventStreamOptions) {
+ if (!opts || !$isCallable(opts.start)) throw new TypeError("EventStream requires an object with `start`");
+ var queue: any[] = [];
super({
type: "direct",
pull: controller => {
this.#ctrl = controller;
- opts?.start?.(this);
+ if (queue.length) {
+ for (const item of queue) {
+ controller.write(item);
+ if (item === null) {
+ controller.close();
+ this.#ctrl = null;
+ return;
+ }
+ }
+ controller.flush();
+ }
+ opts.start?.(this);
},
cancel: () => {
- opts?.cancel?.(this);
- this.#ctrl = undefined;
+ opts.cancel?.(this);
+ this.#ctrl = null;
},
});
- $putByIdDirectPrivate(this, "contentType", "text/event-stream");
+ this.#ctrl = {
+ write: buf => queue.push(buf),
+ flush: () => {},
+ close: () => {
+ queue.push(null);
+ },
+ } as any;
}
- send(event?: unknown, data?: unknown): void {
+ setReconnectionTime(time: number): void {
var ctrl = this.#ctrl!;
if (!ctrl) {
throw new Error("EventStream is closed");
}
+ ctrl.write("retry:" + time + "\n\n");
+ }
+
+ send(event?: unknown, data?: unknown, id?: number | null | undefined): void {
+ var ctrl = this.#ctrl;
+ if (!ctrl) {
+ throw new Error("EventStream is closed");
+ }
if (!data) {
data = event;
event = undefined;
@@ -35,17 +68,51 @@ export function getEventStream() {
if (data === undefined) {
throw new TypeError("EventStream.send() requires a data argument");
}
- if (typeof data === "string") {
- ctrl.write("data: " + data.replace(/\n/g, "\ndata: ") + "\n\n");
+ if (id !== undefined) {
+ this.#writeEventWithId(ctrl, event, data, id);
} else {
- if (event) ctrl.write("event: " + event + "\n");
- ctrl.write("data: " + JSON.stringify(data) + "\n\n");
+ this.#writeEvent(ctrl, event, data);
}
ctrl.flush();
}
+ #writeEvent(ctrl: ReadableStreamDirectController, event: unknown, data: unknown) {
+ if (event) ctrl.write("event:" + event + "\n");
+ if (typeof data === "string") {
+ ctrl.write("data:" + data.replace(/\n/g, "\ndata:") + "\n\n");
+ } else if ($isTypedArrayView(data) || data instanceof ArrayBuffer) {
+ // TODO: handle newlines in this buffer
+ ctrl.write("data:");
+ ctrl.write(data as BufferSource);
+ ctrl.write("\n\n");
+ } else {
+ ctrl.write("data:" + JSON.stringify(data) + "\n\n");
+ }
+ }
+
+ #writeEventWithId(ctrl: ReadableStreamDirectController, event: unknown, data: unknown, id: number | null) {
+ if (event) ctrl.write("event:" + event + "\n");
+ if (typeof data === "string") {
+ ctrl.write("data:" + data.replace(/\n/g, "\ndata:") + "\n");
+ } else if (data instanceof Uint8Array) {
+ // TODO: handle newlines in this buffer
+ ctrl.write("data:");
+ ctrl.write(data);
+ ctrl.write("\n");
+ } else {
+ ctrl.write("data:" + JSON.stringify(data) + "\n");
+ }
+ if (id === null) {
+ ctrl.write("id\n");
+ } else {
+ ctrl.write("id:" + id + "\n");
+ }
+ }
+
close() {
this.#ctrl?.close();
}
- };
+ }
+ Object.defineProperty(EventStream, "name", { value: "EventStream" });
+ return EventStream;
}