aboutsummaryrefslogtreecommitdiff
path: root/src/js/builtins/EventStream.ts
diff options
context:
space:
mode:
authorGravatar dave caruso <me@paperdave.net> 2023-06-28 16:00:24 -0400
committerGravatar dave caruso <me@paperdave.net> 2023-06-29 23:40:47 -0400
commitf0a3559e93a87fdb2d7b4c88edf60726fcce0265 (patch)
treed5d69b46429e032aa3a89181f41e4e2cc2a9422a /src/js/builtins/EventStream.ts
parentbb697687467b33b47f16d33d2dc8dfcc6ccf7274 (diff)
downloadbun-f0a3559e93a87fdb2d7b4c88edf60726fcce0265.tar.gz
bun-f0a3559e93a87fdb2d7b4c88edf60726fcce0265.tar.zst
bun-f0a3559e93a87fdb2d7b4c88edf60726fcce0265.zip
add a workaround for eventstream headers
i dont really like it
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;
}