std/http/server_sent_event_stream.ts
Ian Bull 3c2ac9fe65
refactor(http): align additional error messages (#5791)
Co-authored-by: Yoshiya Hinosawa <stibium121@gmail.com>
2024-08-26 14:05:00 +09:00

106 lines
2.9 KiB
TypeScript

// Copyright 2018-2024 the Deno authors. All rights reserved. MIT license.
// This module is browser compatible.
const NEWLINE_REGEXP = /\r\n|\r|\n/;
const encoder = new TextEncoder();
/**
* Represents a message in the Server-Sent Event (SSE) protocol.
*
* @see {@link https://developer.mozilla.org/en-US/docs/Web/API/Server-sent_events/Using_server-sent_events#fields}
*/
export interface ServerSentEventMessage {
/** Ignored by the client. */
comment?: string;
/** A string identifying the type of event described. */
event?: string;
/** The data field for the message. Split by new lines. */
data?: string;
/** The event ID to set the {@linkcode EventSource} object's last event ID value. */
id?: string | number;
/** The reconnection time. */
retry?: number;
}
function assertHasNoNewline(value: string, varName: string, errPrefix: string) {
if (value.match(NEWLINE_REGEXP) !== null) {
throw new SyntaxError(
`${errPrefix}: ${varName} cannot contain a newline`,
);
}
}
/**
* Converts a server-sent message object into a string for the client.
*
* @see {@link https://developer.mozilla.org/en-US/docs/Web/API/Server-sent_events/Using_server-sent_events#event_stream_format}
*/
function stringify(message: ServerSentEventMessage): Uint8Array {
const lines = [];
if (message.comment) {
assertHasNoNewline(
message.comment,
"`message.comment`",
"Cannot serialize message",
);
lines.push(`:${message.comment}`);
}
if (message.event) {
assertHasNoNewline(
message.event,
"`message.event`",
"Cannot serialize message",
);
lines.push(`event:${message.event}`);
}
if (message.data) {
message.data.split(NEWLINE_REGEXP).forEach((line) =>
lines.push(`data:${line}`)
);
}
if (message.id) {
assertHasNoNewline(
message.id.toString(),
"`message.id`",
"Cannot serialize message",
);
lines.push(`id:${message.id}`);
}
if (message.retry) lines.push(`retry:${message.retry}`);
return encoder.encode(lines.join("\n") + "\n\n");
}
/**
* Transforms server-sent message objects into strings for the client.
*
* @see {@link https://developer.mozilla.org/en-US/docs/Web/API/Server-sent_events/Using_server-sent_events}
*
* @example Usage
* ```ts no-assert
* import {
* type ServerSentEventMessage,
* ServerSentEventStream,
* } from "@std/http/server-sent-event-stream";
*
* const stream = ReadableStream.from<ServerSentEventMessage>([
* { data: "hello there" }
* ]).pipeThrough(new ServerSentEventStream());
* new Response(stream, {
* headers: {
* "content-type": "text/event-stream",
* "cache-control": "no-cache",
* },
* });
* ```
*/
export class ServerSentEventStream
extends TransformStream<ServerSentEventMessage, Uint8Array> {
constructor() {
super({
transform: (message, controller) => {
controller.enqueue(stringify(message));
},
});
}
}