2020-09-05 16:39:25 +02:00
|
|
|
// Copyright 2018-2020 the Deno authors. All rights reserved. MIT license.
|
|
|
|
|
|
|
|
((window) => {
|
2020-09-16 22:22:43 +02:00
|
|
|
const core = window.Deno.core;
|
2020-11-10 05:34:42 +02:00
|
|
|
const { requiredArguments, defineEventHandler } = window.__bootstrap.webUtil;
|
2020-09-05 16:39:25 +02:00
|
|
|
const CONNECTING = 0;
|
|
|
|
const OPEN = 1;
|
|
|
|
const CLOSING = 2;
|
|
|
|
const CLOSED = 3;
|
|
|
|
|
|
|
|
class WebSocket extends EventTarget {
|
|
|
|
#readyState = CONNECTING;
|
|
|
|
|
|
|
|
constructor(url, protocols = []) {
|
|
|
|
super();
|
|
|
|
requiredArguments("WebSocket", arguments.length, 1);
|
|
|
|
|
|
|
|
const wsURL = new URL(url);
|
|
|
|
|
|
|
|
if (wsURL.protocol !== "ws:" && wsURL.protocol !== "wss:") {
|
|
|
|
throw new DOMException(
|
|
|
|
"Only ws & wss schemes are allowed in a WebSocket URL.",
|
|
|
|
"SyntaxError",
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (wsURL.hash !== "" || wsURL.href.endsWith("#")) {
|
|
|
|
throw new DOMException(
|
|
|
|
"Fragments are not allowed in a WebSocket URL.",
|
|
|
|
"SyntaxError",
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
this.#url = wsURL.href;
|
|
|
|
|
2020-11-25 15:17:46 +01:00
|
|
|
core.jsonOpSync("op_ws_check_permission", {
|
|
|
|
url: this.#url,
|
|
|
|
});
|
|
|
|
|
2020-09-05 16:39:25 +02:00
|
|
|
if (protocols && typeof protocols === "string") {
|
|
|
|
protocols = [protocols];
|
|
|
|
}
|
|
|
|
|
|
|
|
if (
|
|
|
|
protocols.some((x) => protocols.indexOf(x) !== protocols.lastIndexOf(x))
|
|
|
|
) {
|
|
|
|
throw new DOMException(
|
|
|
|
"Can't supply multiple times the same protocol.",
|
|
|
|
"SyntaxError",
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2020-09-16 22:22:43 +02:00
|
|
|
core.jsonOpAsync("op_ws_create", {
|
2020-09-05 16:39:25 +02:00
|
|
|
url: wsURL.href,
|
2020-10-22 17:09:44 +02:00
|
|
|
protocols: protocols.join(", "),
|
2020-09-05 16:39:25 +02:00
|
|
|
}).then((create) => {
|
|
|
|
if (create.success) {
|
|
|
|
this.#rid = create.rid;
|
|
|
|
this.#extensions = create.extensions;
|
|
|
|
this.#protocol = create.protocol;
|
|
|
|
|
|
|
|
if (this.#readyState === CLOSING) {
|
2020-09-16 22:22:43 +02:00
|
|
|
core.jsonOpAsync("op_ws_close", {
|
2020-09-05 16:39:25 +02:00
|
|
|
rid: this.#rid,
|
|
|
|
}).then(() => {
|
|
|
|
this.#readyState = CLOSED;
|
|
|
|
|
2020-10-27 09:22:03 +11:00
|
|
|
const errEvent = new ErrorEvent("error");
|
2020-09-05 16:39:25 +02:00
|
|
|
errEvent.target = this;
|
|
|
|
this.dispatchEvent(errEvent);
|
|
|
|
|
|
|
|
const event = new CloseEvent("close");
|
|
|
|
event.target = this;
|
|
|
|
this.dispatchEvent(event);
|
2020-09-17 18:09:50 +02:00
|
|
|
core.close(this.#rid);
|
2020-09-05 16:39:25 +02:00
|
|
|
});
|
|
|
|
} else {
|
|
|
|
this.#readyState = OPEN;
|
|
|
|
const event = new Event("open");
|
|
|
|
event.target = this;
|
|
|
|
this.dispatchEvent(event);
|
|
|
|
|
|
|
|
this.#eventLoop();
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
this.#readyState = CLOSED;
|
|
|
|
|
2020-10-27 09:22:03 +11:00
|
|
|
const errEvent = new ErrorEvent("error");
|
2020-09-05 16:39:25 +02:00
|
|
|
errEvent.target = this;
|
|
|
|
this.dispatchEvent(errEvent);
|
|
|
|
|
|
|
|
const closeEvent = new CloseEvent("close");
|
|
|
|
closeEvent.target = this;
|
|
|
|
this.dispatchEvent(closeEvent);
|
|
|
|
}
|
2020-09-13 11:52:20 +02:00
|
|
|
}).catch((err) => {
|
2020-09-29 11:42:29 +02:00
|
|
|
this.#readyState = CLOSED;
|
|
|
|
|
|
|
|
const errorEv = new ErrorEvent(
|
2020-09-13 11:52:20 +02:00
|
|
|
"error",
|
|
|
|
{ error: err, message: err.toString() },
|
|
|
|
);
|
2020-09-29 11:42:29 +02:00
|
|
|
errorEv.target = this;
|
|
|
|
this.dispatchEvent(errorEv);
|
|
|
|
|
|
|
|
const closeEv = new CloseEvent("close");
|
|
|
|
closeEv.target = this;
|
|
|
|
this.dispatchEvent(closeEv);
|
2020-09-05 16:39:25 +02:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
get CONNECTING() {
|
|
|
|
return CONNECTING;
|
|
|
|
}
|
|
|
|
get OPEN() {
|
|
|
|
return OPEN;
|
|
|
|
}
|
|
|
|
get CLOSING() {
|
|
|
|
return CLOSING;
|
|
|
|
}
|
|
|
|
get CLOSED() {
|
|
|
|
return CLOSED;
|
|
|
|
}
|
|
|
|
|
|
|
|
get readyState() {
|
|
|
|
return this.#readyState;
|
|
|
|
}
|
|
|
|
|
|
|
|
#extensions = "";
|
|
|
|
#protocol = "";
|
|
|
|
#url = "";
|
|
|
|
#rid;
|
|
|
|
|
|
|
|
get extensions() {
|
|
|
|
return this.#extensions;
|
|
|
|
}
|
|
|
|
get protocol() {
|
|
|
|
return this.#protocol;
|
|
|
|
}
|
|
|
|
|
|
|
|
#binaryType = "blob";
|
|
|
|
get binaryType() {
|
|
|
|
return this.#binaryType;
|
|
|
|
}
|
|
|
|
set binaryType(value) {
|
|
|
|
if (value === "blob" || value === "arraybuffer") {
|
|
|
|
this.#binaryType = value;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#bufferedAmount = 0;
|
|
|
|
get bufferedAmount() {
|
|
|
|
return this.#bufferedAmount;
|
|
|
|
}
|
|
|
|
|
|
|
|
get url() {
|
|
|
|
return this.#url;
|
|
|
|
}
|
|
|
|
|
|
|
|
send(data) {
|
|
|
|
requiredArguments("WebSocket.send", arguments.length, 1);
|
|
|
|
|
|
|
|
if (this.#readyState != OPEN) {
|
|
|
|
throw Error("readyState not OPEN");
|
|
|
|
}
|
|
|
|
|
|
|
|
const sendTypedArray = (ta) => {
|
|
|
|
this.#bufferedAmount += ta.size;
|
2020-09-16 22:22:43 +02:00
|
|
|
core.jsonOpAsync("op_ws_send", {
|
2020-09-05 16:39:25 +02:00
|
|
|
rid: this.#rid,
|
2021-01-05 13:37:02 +01:00
|
|
|
kind: "binary",
|
2020-09-05 16:39:25 +02:00
|
|
|
}, ta).then(() => {
|
|
|
|
this.#bufferedAmount -= ta.size;
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
|
|
|
if (data instanceof Blob) {
|
|
|
|
data.slice().arrayBuffer().then((ab) =>
|
|
|
|
sendTypedArray(new DataView(ab))
|
|
|
|
);
|
|
|
|
} else if (
|
|
|
|
data instanceof Int8Array || data instanceof Int16Array ||
|
|
|
|
data instanceof Int32Array || data instanceof Uint8Array ||
|
|
|
|
data instanceof Uint16Array || data instanceof Uint32Array ||
|
|
|
|
data instanceof Uint8ClampedArray || data instanceof Float32Array ||
|
|
|
|
data instanceof Float64Array || data instanceof DataView
|
|
|
|
) {
|
|
|
|
sendTypedArray(data);
|
|
|
|
} else if (data instanceof ArrayBuffer) {
|
|
|
|
sendTypedArray(new DataView(data));
|
|
|
|
} else {
|
|
|
|
const string = String(data);
|
|
|
|
const encoder = new TextEncoder();
|
|
|
|
const d = encoder.encode(string);
|
|
|
|
this.#bufferedAmount += d.size;
|
2020-09-16 22:22:43 +02:00
|
|
|
core.jsonOpAsync("op_ws_send", {
|
2020-09-05 16:39:25 +02:00
|
|
|
rid: this.#rid,
|
2021-01-05 13:37:02 +01:00
|
|
|
kind: "text",
|
2020-09-05 16:39:25 +02:00
|
|
|
text: string,
|
|
|
|
}).then(() => {
|
|
|
|
this.#bufferedAmount -= d.size;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
close(code, reason) {
|
|
|
|
if (code && (code !== 1000 && !(3000 <= code > 5000))) {
|
|
|
|
throw new DOMException(
|
|
|
|
"The close code must be either 1000 or in the range of 3000 to 4999.",
|
|
|
|
"NotSupportedError",
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
const encoder = new TextEncoder();
|
|
|
|
if (reason && encoder.encode(reason).byteLength > 123) {
|
|
|
|
throw new DOMException(
|
|
|
|
"The close reason may not be longer than 123 bytes.",
|
|
|
|
"SyntaxError",
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (this.#readyState === CONNECTING) {
|
|
|
|
this.#readyState = CLOSING;
|
|
|
|
} else if (this.#readyState === OPEN) {
|
|
|
|
this.#readyState = CLOSING;
|
|
|
|
|
2020-09-16 22:22:43 +02:00
|
|
|
core.jsonOpAsync("op_ws_close", {
|
2020-09-05 16:39:25 +02:00
|
|
|
rid: this.#rid,
|
|
|
|
code,
|
|
|
|
reason,
|
|
|
|
}).then(() => {
|
|
|
|
this.#readyState = CLOSED;
|
|
|
|
const event = new CloseEvent("close", {
|
|
|
|
wasClean: true,
|
|
|
|
code,
|
|
|
|
reason,
|
|
|
|
});
|
|
|
|
event.target = this;
|
|
|
|
this.dispatchEvent(event);
|
2020-09-17 18:09:50 +02:00
|
|
|
core.close(this.#rid);
|
2020-09-05 16:39:25 +02:00
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
async #eventLoop() {
|
|
|
|
if (this.#readyState === OPEN) {
|
2020-09-16 22:22:43 +02:00
|
|
|
const message = await core.jsonOpAsync(
|
|
|
|
"op_ws_next_event",
|
|
|
|
{ rid: this.#rid },
|
|
|
|
);
|
2020-09-05 16:39:25 +02:00
|
|
|
if (message.type === "string" || message.type === "binary") {
|
|
|
|
let data;
|
|
|
|
|
|
|
|
if (message.type === "string") {
|
|
|
|
data = message.data;
|
|
|
|
} else {
|
|
|
|
if (this.binaryType === "blob") {
|
|
|
|
data = new Blob([new Uint8Array(message.data)]);
|
|
|
|
} else {
|
|
|
|
data = new Uint8Array(message.data).buffer;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const event = new MessageEvent("message", {
|
|
|
|
data,
|
|
|
|
origin: this.#url,
|
|
|
|
});
|
|
|
|
event.target = this;
|
|
|
|
this.dispatchEvent(event);
|
|
|
|
|
2021-01-05 13:37:02 +01:00
|
|
|
this.#eventLoop();
|
|
|
|
} else if (message.type === "ping") {
|
|
|
|
core.jsonOpAsync("op_ws_send", {
|
|
|
|
rid: this.#rid,
|
|
|
|
kind: "pong",
|
|
|
|
});
|
|
|
|
|
2020-09-05 16:39:25 +02:00
|
|
|
this.#eventLoop();
|
|
|
|
} else if (message.type === "close") {
|
|
|
|
this.#readyState = CLOSED;
|
|
|
|
const event = new CloseEvent("close", {
|
|
|
|
wasClean: true,
|
|
|
|
code: message.code,
|
|
|
|
reason: message.reason,
|
|
|
|
});
|
|
|
|
event.target = this;
|
|
|
|
this.dispatchEvent(event);
|
|
|
|
} else if (message.type === "error") {
|
2020-09-29 11:42:29 +02:00
|
|
|
this.#readyState = CLOSED;
|
|
|
|
|
2020-10-27 09:22:03 +11:00
|
|
|
const errorEv = new ErrorEvent("error");
|
2020-09-29 11:42:29 +02:00
|
|
|
errorEv.target = this;
|
|
|
|
this.dispatchEvent(errorEv);
|
|
|
|
|
|
|
|
this.#readyState = CLOSED;
|
|
|
|
const closeEv = new CloseEvent("close");
|
|
|
|
closeEv.target = this;
|
|
|
|
this.dispatchEvent(closeEv);
|
2020-09-05 16:39:25 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Object.defineProperties(WebSocket, {
|
|
|
|
CONNECTING: {
|
|
|
|
value: 0,
|
|
|
|
},
|
|
|
|
OPEN: {
|
|
|
|
value: 1,
|
|
|
|
},
|
|
|
|
CLOSING: {
|
|
|
|
value: 2,
|
|
|
|
},
|
|
|
|
CLOSED: {
|
|
|
|
value: 3,
|
|
|
|
},
|
|
|
|
});
|
|
|
|
|
2020-11-10 05:34:42 +02:00
|
|
|
defineEventHandler(WebSocket.prototype, "message");
|
|
|
|
defineEventHandler(WebSocket.prototype, "error");
|
|
|
|
defineEventHandler(WebSocket.prototype, "close");
|
|
|
|
defineEventHandler(WebSocket.prototype, "open");
|
2020-09-05 16:39:25 +02:00
|
|
|
window.__bootstrap.webSocket = {
|
|
|
|
WebSocket,
|
|
|
|
};
|
|
|
|
})(this);
|