mirror of
https://github.com/denoland/deno.git
synced 2024-11-23 15:16:54 -05:00
5c1fa0cf9c
Depends on #22493. Closes #22367.
383 lines
9.7 KiB
JavaScript
383 lines
9.7 KiB
JavaScript
// Copyright 2018-2024 the Deno authors. All rights reserved. MIT license.
|
|
|
|
/// <reference path="../../core/internal.d.ts" />
|
|
|
|
import { primordials } from "ext:core/mod.js";
|
|
import { op_utf8_to_byte_string } from "ext:core/ops";
|
|
const {
|
|
ArrayPrototypeFind,
|
|
Number,
|
|
NumberIsFinite,
|
|
NumberIsNaN,
|
|
ObjectDefineProperties,
|
|
ObjectPrototypeIsPrototypeOf,
|
|
StringPrototypeEndsWith,
|
|
StringPrototypeIncludes,
|
|
StringPrototypeIndexOf,
|
|
StringPrototypeSlice,
|
|
StringPrototypeStartsWith,
|
|
StringPrototypeToLowerCase,
|
|
SymbolFor,
|
|
} = primordials;
|
|
|
|
import * as webidl from "ext:deno_webidl/00_webidl.js";
|
|
import { createFilteredInspectProxy } from "ext:deno_console/01_console.js";
|
|
import { URL } from "ext:deno_url/00_url.js";
|
|
import { DOMException } from "ext:deno_web/01_dom_exception.js";
|
|
import {
|
|
defineEventHandler,
|
|
EventTarget,
|
|
setIsTrusted,
|
|
} from "ext:deno_web/02_event.js";
|
|
import { clearTimeout, setTimeout } from "ext:deno_web/02_timers.js";
|
|
import { TransformStream } from "ext:deno_web/06_streams.js";
|
|
import { TextDecoderStream } from "ext:deno_web/08_text_encoding.js";
|
|
import { getLocationHref } from "ext:deno_web/12_location.js";
|
|
import { newInnerRequest } from "ext:deno_fetch/23_request.js";
|
|
import { mainFetch } from "ext:deno_fetch/26_fetch.js";
|
|
|
|
// Copied from https://github.com/denoland/deno_std/blob/e0753abe0c8602552862a568348c046996709521/streams/text_line_stream.ts#L20-L74
|
|
export class TextLineStream extends TransformStream {
|
|
#allowCR;
|
|
#buf = "";
|
|
|
|
constructor(options) {
|
|
super({
|
|
transform: (chunk, controller) => this.#handle(chunk, controller),
|
|
flush: (controller) => {
|
|
if (this.#buf.length > 0) {
|
|
if (
|
|
this.#allowCR &&
|
|
this.#buf[this.#buf.length - 1] === "\r"
|
|
) controller.enqueue(StringPrototypeSlice(this.#buf, 0, -1));
|
|
else controller.enqueue(this.#buf);
|
|
}
|
|
},
|
|
});
|
|
this.#allowCR = options?.allowCR ?? false;
|
|
}
|
|
|
|
#handle(chunk, controller) {
|
|
chunk = this.#buf + chunk;
|
|
|
|
for (;;) {
|
|
const lfIndex = StringPrototypeIndexOf(chunk, "\n");
|
|
|
|
if (this.#allowCR) {
|
|
const crIndex = StringPrototypeIndexOf(chunk, "\r");
|
|
|
|
if (
|
|
crIndex !== -1 && crIndex !== (chunk.length - 1) &&
|
|
(lfIndex === -1 || (lfIndex - 1) > crIndex)
|
|
) {
|
|
controller.enqueue(StringPrototypeSlice(chunk, 0, crIndex));
|
|
chunk = StringPrototypeSlice(chunk, crIndex + 1);
|
|
continue;
|
|
}
|
|
}
|
|
|
|
if (lfIndex !== -1) {
|
|
let crOrLfIndex = lfIndex;
|
|
if (chunk[lfIndex - 1] === "\r") {
|
|
crOrLfIndex--;
|
|
}
|
|
controller.enqueue(StringPrototypeSlice(chunk, 0, crOrLfIndex));
|
|
chunk = StringPrototypeSlice(chunk, lfIndex + 1);
|
|
continue;
|
|
}
|
|
|
|
break;
|
|
}
|
|
|
|
this.#buf = chunk;
|
|
}
|
|
}
|
|
|
|
const CONNECTING = 0;
|
|
const OPEN = 1;
|
|
const CLOSED = 2;
|
|
|
|
class EventSource extends EventTarget {
|
|
/** @type {AbortController} */
|
|
#abortController = new AbortController();
|
|
|
|
/** @type {number | undefined} */
|
|
#reconnectionTimerId;
|
|
|
|
/** @type {number} */
|
|
#reconnectionTime = 5000;
|
|
|
|
/** @type {string} */
|
|
#lastEventId = "";
|
|
|
|
/** @type {number} */
|
|
#readyState = CONNECTING;
|
|
get readyState() {
|
|
webidl.assertBranded(this, EventSourcePrototype);
|
|
return this.#readyState;
|
|
}
|
|
|
|
get CONNECTING() {
|
|
webidl.assertBranded(this, EventSourcePrototype);
|
|
return CONNECTING;
|
|
}
|
|
get OPEN() {
|
|
webidl.assertBranded(this, EventSourcePrototype);
|
|
return OPEN;
|
|
}
|
|
get CLOSED() {
|
|
webidl.assertBranded(this, EventSourcePrototype);
|
|
return CLOSED;
|
|
}
|
|
|
|
/** @type {string} */
|
|
#url;
|
|
get url() {
|
|
webidl.assertBranded(this, EventSourcePrototype);
|
|
return this.#url;
|
|
}
|
|
|
|
/** @type {boolean} */
|
|
#withCredentials;
|
|
get withCredentials() {
|
|
webidl.assertBranded(this, EventSourcePrototype);
|
|
return this.#withCredentials;
|
|
}
|
|
|
|
constructor(url, eventSourceInitDict = {}) {
|
|
super();
|
|
this[webidl.brand] = webidl.brand;
|
|
const prefix = "Failed to construct 'EventSource'";
|
|
webidl.requiredArguments(arguments.length, 1, prefix);
|
|
url = webidl.converters.USVString(url, prefix, "Argument 1");
|
|
eventSourceInitDict = webidl.converters.EventSourceInit(
|
|
eventSourceInitDict,
|
|
prefix,
|
|
"Argument 2",
|
|
);
|
|
|
|
try {
|
|
url = new URL(url, getLocationHref()).href;
|
|
} catch (e) {
|
|
throw new DOMException(e.message, "SyntaxError");
|
|
}
|
|
|
|
this.#url = url;
|
|
this.#withCredentials = eventSourceInitDict.withCredentials;
|
|
|
|
this.#loop();
|
|
}
|
|
|
|
close() {
|
|
webidl.assertBranded(this, EventSourcePrototype);
|
|
this.#abortController.abort();
|
|
this.#readyState = CLOSED;
|
|
clearTimeout(this.#reconnectionTimerId);
|
|
}
|
|
|
|
async #loop() {
|
|
const lastEventIdValue = this.#lastEventId;
|
|
const req = newInnerRequest(
|
|
"GET",
|
|
this.#url,
|
|
() =>
|
|
lastEventIdValue === ""
|
|
? [
|
|
["accept", "text/event-stream"],
|
|
]
|
|
: [
|
|
["accept", "text/event-stream"],
|
|
["Last-Event-Id", op_utf8_to_byte_string(lastEventIdValue)],
|
|
],
|
|
null,
|
|
false,
|
|
);
|
|
/** @type {InnerResponse} */
|
|
let res;
|
|
try {
|
|
res = await mainFetch(req, true, this.#abortController.signal);
|
|
} catch {
|
|
this.#reestablishConnection();
|
|
return;
|
|
}
|
|
|
|
if (res.aborted) {
|
|
this.#failConnection();
|
|
return;
|
|
}
|
|
if (res.type === "error") {
|
|
this.#reestablishConnection();
|
|
return;
|
|
}
|
|
const contentType = ArrayPrototypeFind(
|
|
res.headerList,
|
|
(header) => StringPrototypeToLowerCase(header[0]) === "content-type",
|
|
);
|
|
if (
|
|
res.status !== 200 ||
|
|
!contentType ||
|
|
!StringPrototypeIncludes(
|
|
StringPrototypeToLowerCase(contentType[1]),
|
|
"text/event-stream",
|
|
)
|
|
) {
|
|
this.#failConnection();
|
|
return;
|
|
}
|
|
|
|
if (this.#readyState === CLOSED) {
|
|
return;
|
|
}
|
|
this.#readyState = OPEN;
|
|
this.dispatchEvent(new Event("open"));
|
|
|
|
let data = "";
|
|
let eventType = "";
|
|
let lastEventId = this.#lastEventId;
|
|
|
|
try {
|
|
for await (
|
|
// deno-lint-ignore prefer-primordials
|
|
const chunk of res.body.stream
|
|
.pipeThrough(new TextDecoderStream())
|
|
.pipeThrough(new TextLineStream({ allowCR: true }))
|
|
) {
|
|
if (chunk === "") {
|
|
this.#lastEventId = lastEventId;
|
|
if (data === "") {
|
|
eventType = "";
|
|
continue;
|
|
}
|
|
if (StringPrototypeEndsWith(data, "\n")) {
|
|
data = StringPrototypeSlice(data, 0, -1);
|
|
}
|
|
const event = new MessageEvent(eventType || "message", {
|
|
data,
|
|
origin: res.url(),
|
|
lastEventId: this.#lastEventId,
|
|
});
|
|
setIsTrusted(event, true);
|
|
data = "";
|
|
eventType = "";
|
|
if (this.#readyState !== CLOSED) {
|
|
this.dispatchEvent(event);
|
|
}
|
|
} else if (StringPrototypeStartsWith(chunk, ":")) {
|
|
continue;
|
|
} else {
|
|
let field = chunk;
|
|
let value = "";
|
|
const colonIndex = StringPrototypeIndexOf(chunk, ":");
|
|
if (colonIndex !== -1) {
|
|
field = StringPrototypeSlice(chunk, 0, colonIndex);
|
|
value = StringPrototypeSlice(chunk, colonIndex + 1);
|
|
if (StringPrototypeStartsWith(value, " ")) {
|
|
value = StringPrototypeSlice(value, 1);
|
|
}
|
|
}
|
|
|
|
switch (field) {
|
|
case "event": {
|
|
eventType = value;
|
|
break;
|
|
}
|
|
case "data": {
|
|
data += value + "\n";
|
|
break;
|
|
}
|
|
case "id": {
|
|
if (!StringPrototypeIncludes(value, "\0")) {
|
|
lastEventId = value;
|
|
}
|
|
break;
|
|
}
|
|
case "retry": {
|
|
const reconnectionTime = Number(value);
|
|
if (
|
|
!NumberIsNaN(reconnectionTime) &&
|
|
NumberIsFinite(reconnectionTime)
|
|
) {
|
|
this.#reconnectionTime = reconnectionTime;
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
} catch {
|
|
// The connection is reestablished below
|
|
}
|
|
|
|
this.#reestablishConnection();
|
|
}
|
|
|
|
#reestablishConnection() {
|
|
if (this.#readyState === CLOSED) {
|
|
return;
|
|
}
|
|
this.#readyState = CONNECTING;
|
|
this.dispatchEvent(new Event("error"));
|
|
this.#reconnectionTimerId = setTimeout(() => {
|
|
if (this.#readyState !== CONNECTING) {
|
|
return;
|
|
}
|
|
this.#loop();
|
|
}, this.#reconnectionTime);
|
|
}
|
|
|
|
#failConnection() {
|
|
if (this.#readyState !== CLOSED) {
|
|
this.#readyState = CLOSED;
|
|
this.dispatchEvent(new Event("error"));
|
|
}
|
|
}
|
|
|
|
[SymbolFor("Deno.privateCustomInspect")](inspect, inspectOptions) {
|
|
return inspect(
|
|
createFilteredInspectProxy({
|
|
object: this,
|
|
evaluate: ObjectPrototypeIsPrototypeOf(EventSourcePrototype, this),
|
|
keys: [
|
|
"readyState",
|
|
"url",
|
|
"withCredentials",
|
|
"onopen",
|
|
"onmessage",
|
|
"onerror",
|
|
],
|
|
}),
|
|
inspectOptions,
|
|
);
|
|
}
|
|
}
|
|
|
|
const EventSourcePrototype = EventSource.prototype;
|
|
|
|
ObjectDefineProperties(EventSource, {
|
|
CONNECTING: {
|
|
value: 0,
|
|
},
|
|
OPEN: {
|
|
value: 1,
|
|
},
|
|
CLOSED: {
|
|
value: 2,
|
|
},
|
|
});
|
|
|
|
defineEventHandler(EventSource.prototype, "open");
|
|
defineEventHandler(EventSource.prototype, "message");
|
|
defineEventHandler(EventSource.prototype, "error");
|
|
|
|
webidl.converters.EventSourceInit = webidl.createDictionaryConverter(
|
|
"EventSourceInit",
|
|
[
|
|
{
|
|
key: "withCredentials",
|
|
defaultValue: false,
|
|
converter: webidl.converters.boolean,
|
|
},
|
|
],
|
|
);
|
|
|
|
export { EventSource };
|