diff --git a/docs/godslayerakp/ws.md b/docs/godslayerakp/ws.md new file mode 100644 index 0000000000..e9b09a01da --- /dev/null +++ b/docs/godslayerakp/ws.md @@ -0,0 +1,127 @@ +# WebSocket + +This extension lets you communicate directly with most [WebSocket](https://en.wikipedia.org/wiki/WebSocket) servers. This is the protocol that things like cloud variables and Cloudlink use. + +These are rather low level blocks. They let you establish the connection, but your project still needs to know what kinds of messages to send and how to read messages from the server. + +## Blocks + +```scratch +connect to [wss://...] :: #307eff +``` +You have to run this block before any of the other blocks can do anything. You need to provide a valid WebSocket URL. + +The URL should start with `ws://` or `wss://`. For security reasons, `ws://` URLs will usually only work if the WebSocket is running on your computer (for example, `ws://localhost:8000`). + +Something simple to play with is the echo server: `wss://echoserver.redman13.repl.co`. Any message you send to it, it'll send right back to you. + +Note that connections are **per sprite**. Each sprite (or clone) can connect to one server at a time. Multiple sprites can connect to the same or different servers as much as your computer allows, but note those will all be separate connections. + +--- + +```scratch +when connected :: hat #307eff +``` +
+ +```scratch + +``` +Connecting to the server can take some time. Use these blocks to know when the connection was successful. After this, you can start sending and receiving messages. + +When the connection is lost, any blocks under the hat will also be stopped. + +--- + +```scratch +when message received :: hat #307eff +``` +
+ +```scratch +(received message data :: #307eff) +``` + +These blocks let you receive messages from the server. The hat block block will run once for each message the server sends with the data stored in the round reporter block. + +Note that WebSocket supports two types of messages: + + - **Text messages**: The data in the block will just be the raw text from the server. + - **Binary messages**: The data in the block will be a base64-encoded data: URL of the data, as it may not be safe to store directly in a string. You can use other extensions to convert this to something useful, such as fetch, depending on what data it contains. + +If multiple messages are received in a single frame or if your message processing logic causes delays (for example, using wait blocks), messages after the first one will be placed in a **queue**. Once your script finishes, if there's anything in the queue, the "when message received" block will run again the next frame. + +--- + +```scratch +send message (...) :: #307eff +``` + +This is the other side: it lets you send messages to the server. Only text messages are supported; binary messages are not yet supported. + +There's no queue this time. The messages are sent over as fast as your internet connection and the server will allow. + +--- + +```scratch +when connection closes :: hat #307eff +``` +
+ +```scratch + +``` +
+ +These let you detect when either the server closes the connection or your project closes the connection. They don't distinguish. Note that connections have separate blocks. + +Servers can close connections for a lot of reasons: perhaps it's restarting, or perhaps your project tried to do something the server didn't like. + +```scratch +(closing code :: #307eff) +``` +
+ +```scratch +(closing message :: #307eff) +``` + +These blocks can help you gain some insight. Closing code is a number from the WebSocket protocol. There is a [big table](https://developer.mozilla.org/en-US/docs/Web/API/CloseEvent/code#value) of possible values, but generally there is very little to gain from looking at these. + +Servers can also send a text "reason" when they close the connection, although almost no servers actually do this. + +```scratch +close connection :: #307eff +``` +
+ +```scratch +close connection with code (1000) :: #307eff +``` +
+ +```scratch +close connection with reason (...) and code (1000) :: #307eff +``` + +Your project can also close the connection whenever it wants. All of these blocks do basically the same thing. + +Just like how the server can send a code and a reason when it closes the connection, you can send those to the server. Note some limitations: + + - **Code** can be either the number 1000 ("Normal Closure") or an integer in the range 3000-4999 (meaning depends on what server you're talking to). Anything not in this range will be converted to 1000. Few servers will look at this. + - **Reason** can be any text up to 123-bytes long when encoded as UTF-8. Usually that just means up to 123 characters, but things like Emoji are technically multiple characters. Regardless very few servers will even bother to look at this. + +--- + +```scratch +when connection errors :: hat #307eff +``` +
+ +```scratch + +``` + +Sometimes things don't go so well. Maybe your internet connection died, the server is down, or you typed in the wrong URL. There's a lot of things that can go wrong. These let you try to handle that. + +Unfortunately we can't give much insight as to what caused the errors. Your browser tells us very little, but even if it did give us more information, it probably wouldn't be very helpful. diff --git a/extensions/extensions.json b/extensions/extensions.json index 2781ff7955..c6892eb6d7 100644 --- a/extensions/extensions.json +++ b/extensions/extensions.json @@ -73,6 +73,7 @@ "qxsck/var-and-list", "vercte/dictionaries", "godslayerakp/http", + "godslayerakp/ws", "Lily/CommentBlocks", "veggiecan/LongmanDictionary", "CubesterYT/TurboHook", diff --git a/extensions/godslayerakp/ws.js b/extensions/godslayerakp/ws.js new file mode 100644 index 0000000000..bb49c3a876 --- /dev/null +++ b/extensions/godslayerakp/ws.js @@ -0,0 +1,499 @@ +// Name: WebSocket +// ID: gsaWebsocket +// Description: Manually connect to WebSocket servers. +// By: RedMan13 + +(function (Scratch) { + "use strict"; + + if (!Scratch.extensions.unsandboxed) { + throw new Error("can not load outside unsandboxed mode"); + } + + const blobToDataURL = (blob) => + new Promise((resolve, reject) => { + const fr = new FileReader(); + fr.onload = () => resolve(fr.result); + fr.onerror = () => + reject(new Error(`Failed to read as data URL: ${fr.error}`)); + fr.readAsDataURL(blob); + }); + + /* ------- BLOCKS -------- */ + const { BlockType, Cast, ArgumentType } = Scratch; + const vm = Scratch.vm; + const runtime = vm.runtime; + + /** + * @typedef WebSocketInfo + * @property {boolean} instanceReplaced + * @property {boolean} manuallyClosed + * @property {boolean} errored + * @property {string} closeMessage + * @property {number} closeCode + * @property {string} data + * @property {string[]} messageQueue + * @property {WebSocket|null} websocket + * @property {VM.Thread[]} connectThreads + * @property {boolean} messageThreadsRunning + * @property {VM.Thread[]} messageThreads + * @property {object[]} sendOnceConnected + */ + + /** + * @param {unknown} exitCode + * @return {number} a valid code that won't throw an error in WebSocket#close() + */ + const toCloseCode = (exitCode) => { + const casted = Cast.toNumber(exitCode); + // Only valid values are 1000 or the range 3000-4999 + if (casted === 1000 || (casted >= 3000 && casted <= 4999)) { + return casted; + } + return 1000; + }; + + /** + * @param {unknown} reason + * @returns {string} a valid reason that won't throw an error in WebSocket#close() + */ + const toCloseReason = (reason) => { + const casted = Cast.toString(reason); + + // Reason can't be longer than 123 UTF-8 bytes + // We can't just truncate by reason.length as that would not work for eg. emoji + const encoder = new TextEncoder(); + let encoded = encoder.encode(casted); + encoded = encoded.slice(0, 123); + + // Now we have another problem: If the 123 byte cut-off produced invalid UTF-8, we + // need to keep cutting off bytes until it's valid. + const decoder = new TextDecoder(); + while (encoded.byteLength > 0) { + try { + const decoded = decoder.decode(encoded); + return decoded; + } catch (e) { + encoded = encoded.slice(0, encoded.byteLength - 1); + } + } + + return ""; + }; + + class WebSocketExtension { + /** + * no need to install runtime as it comes with Scratch var + */ + constructor() { + /** @type {Record} */ + this.instances = {}; + + runtime.on("targetWasRemoved", (target) => { + const instance = this.instances[target.id]; + if (instance) { + instance.manuallyClosed = true; + if (instance.websocket) { + instance.websocket.close(); + } + delete this.instances[target.id]; + } + }); + } + getInfo() { + return { + id: "gsaWebsocket", + name: "WebSocket", + docsURI: "https://extensions.turbowarp.org/godslayerakp/ws", + color1: "#307eff", + color2: "#2c5eb0", + blocks: [ + { + opcode: "newInstance", + blockType: BlockType.COMMAND, + arguments: { + URL: { + type: ArgumentType.STRING, + defaultValue: "wss://echoserver.redman13.repl.co", + }, + }, + text: "connect to [URL]", + }, + "---", + { + opcode: "onOpen", + blockType: BlockType.EVENT, + isEdgeActivated: false, + shouldRestartExistingThreads: true, + text: "when connected", + }, + { + opcode: "isConnected", + blockType: BlockType.BOOLEAN, + text: "is connected?", + disableMonitor: true, + }, + "---", + { + opcode: "onMessage", + blockType: BlockType.EVENT, + isEdgeActivated: false, + shouldRestartExistingThreads: true, + text: "when message received", + }, + { + opcode: "messageData", + blockType: BlockType.REPORTER, + text: "received message data", + disableMonitor: true, + }, + "---", + { + opcode: "sendMessage", + blockType: BlockType.COMMAND, + arguments: { + PAYLOAD: { + type: ArgumentType.STRING, + defaultValue: "hello!", + }, + }, + text: "send message [PAYLOAD]", + }, + "---", + { + opcode: "onError", + blockType: BlockType.EVENT, + isEdgeActivated: false, + shouldRestartExistingThreads: true, + text: "when connection errors", + }, + { + opcode: "hasErrored", + blockType: BlockType.BOOLEAN, + text: "has connection errored?", + disableMonitor: true, + }, + "---", + { + opcode: "onClose", + blockType: BlockType.EVENT, + isEdgeActivated: false, + shouldRestartExistingThreads: true, + text: "when connection closes", + }, + { + opcode: "isClosed", + blockType: BlockType.BOOLEAN, + text: "is connection closed?", + disableMonitor: true, + }, + { + opcode: "closeCode", + blockType: BlockType.REPORTER, + text: "closing code", + disableMonitor: true, + }, + { + opcode: "closeMessage", + blockType: BlockType.REPORTER, + text: "closing message", + disableMonitor: true, + }, + { + opcode: "closeWithoutReason", + blockType: BlockType.COMMAND, + text: "close connection", + }, + { + opcode: "closeWithCode", + blockType: BlockType.COMMAND, + arguments: { + CODE: { + type: ArgumentType.NUMBER, + defaultValue: "1000", + }, + }, + text: "close connection with code [CODE]", + }, + { + opcode: "closeWithReason", + blockType: BlockType.COMMAND, + arguments: { + CODE: { + type: ArgumentType.NUMBER, + defaultValue: "1000", + }, + REASON: { + type: ArgumentType.STRING, + defaultValue: "fulfilled", + }, + }, + text: "close connection with reason [REASON] and code [CODE]", + }, + ], + }; + } + + newInstance(args, util) { + const target = util.target; + + let url = Cast.toString(args.URL); + if (!/^(ws|wss):/is.test(url)) { + // url doesnt start with a valid connection type + // so we just assume its formated without it + if (/^(?!(ws|http)s?:\/\/).*$/is.test(url)) { + url = `wss://${url}`; + } else if (/^(http|https):/is.test(url)) { + const urlParts = url.split(":"); + urlParts[0] = url.toLowerCase().startsWith("https") ? "wss" : "ws"; + url = urlParts.join(":"); + } else { + // we couldnt fix the url... + return; + } + } + + const oldInstance = this.instances[util.target.id]; + if (oldInstance) { + oldInstance.instanceReplaced = true; + if (oldInstance.websocket) { + oldInstance.websocket.close(); + } + } + + /** @type {WebSocketInfo} */ + const instance = { + instanceReplaced: false, + manuallyClosed: false, + errored: false, + closeMessage: "", + closeCode: 0, + data: "", + websocket: null, + messageThreadsRunning: false, + connectThreads: [], + messageThreads: [], + messageQueue: [], + sendOnceConnected: [], + }; + this.instances[util.target.id] = instance; + + return Scratch.canFetch(url) + .then( + (allowed) => + new Promise((resolve) => { + if ( + !allowed || + instance.instanceReplaced || + instance.manuallyClosed + ) { + resolve(); + return; + } + + // canFetch() checked above + // eslint-disable-next-line no-restricted-syntax + const websocket = new WebSocket(url); + instance.websocket = websocket; + + const beforeExecute = () => { + if (instance.messageThreadsRunning) { + const stillRunning = instance.messageThreads.some((i) => + runtime.isActiveThread(i) + ); + if (!stillRunning) { + const isQueueEmpty = instance.messageQueue.length === 0; + if (isQueueEmpty) { + instance.messageThreadsRunning = false; + instance.messageThreads = []; + } else { + instance.data = instance.messageQueue.shift(); + instance.messageThreads = runtime.startHats( + "gsaWebsocket_onMessage", + null, + target + ); + } + } + } + }; + + const onStopAll = () => { + instance.instanceReplaced = true; + instance.manuallyClosed = true; + instance.websocket.close(); + }; + + vm.runtime.on("BEFORE_EXECUTE", beforeExecute); + vm.runtime.on("PROJECT_STOP_ALL", onStopAll); + + const cleanup = () => { + vm.runtime.off("BEFORE_EXECUTE", beforeExecute); + vm.runtime.off("PROJECT_STOP_ALL", onStopAll); + + for (const thread of instance.connectThreads) { + thread.status = 4; // STATUS_DONE + } + + resolve(); + }; + + websocket.onopen = (e) => { + if (instance.instanceReplaced || instance.manuallyClosed) { + cleanup(); + websocket.close(); + return; + } + + for (const item of instance.sendOnceConnected) { + websocket.send(item); + } + instance.sendOnceConnected.length = 0; + + instance.connectThreads = runtime.startHats( + "gsaWebsocket_onOpen", + null, + target + ); + resolve(); + }; + + websocket.onclose = (e) => { + if (instance.instanceReplaced) return; + instance.closeMessage = e.reason || ""; + instance.closeCode = e.code; + runtime.startHats("gsaWebsocket_onClose", null, target); + cleanup(); + }; + + websocket.onerror = (e) => { + if (instance.instanceReplaced) return; + console.error("websocket error", e); + instance.errored = true; + runtime.startHats("gsaWebsocket_onError", null, target); + cleanup(); + }; + + websocket.onmessage = async (e) => { + if (instance.instanceReplaced || instance.manuallyClosed) + return; + + let data = e.data; + + // Convert binary messages to a data: uri + // TODO: doing this right now might break order? + if (data instanceof Blob) { + data = await blobToDataURL(data); + } + + if (instance.messageThreadsRunning) { + instance.messageQueue.push(data); + } else { + instance.data = data; + instance.messageThreads = runtime.startHats( + "gsaWebsocket_onMessage", + null, + target + ); + instance.messageThreadsRunning = true; + } + }; + }) + ) + .catch((error) => { + console.error("could not open websocket connection", error); + }); + } + + isConnected(_, utils) { + const instance = this.instances[utils.target.id]; + if (!instance) return false; + return ( + !!instance.websocket && instance.websocket.readyState === WebSocket.OPEN + ); + } + + messageData(_, utils) { + const instance = this.instances[utils.target.id]; + if (!instance) return ""; + return instance.data; + } + + isClosed(_, utils) { + const instance = this.instances[utils.target.id]; + if (!instance) return false; + return ( + !!instance.websocket && + instance.websocket.readyState === WebSocket.CLOSED + ); + } + + closeCode(_, utils) { + const instance = this.instances[utils.target.id]; + if (!instance) return 0; + return instance.closeCode; + } + + closeMessage(_, utils) { + const instance = this.instances[utils.target.id]; + if (!instance) return ""; + return instance.closeMessage; + } + + hasErrored(_, utils) { + const instance = this.instances[utils.target.id]; + if (!instance) return false; + return instance.errored; + } + + sendMessage(args, utils) { + const PAYLOAD = Cast.toString(args.PAYLOAD); + const instance = this.instances[utils.target.id]; + if (!instance) return; + + if ( + !instance.websocket || + instance.websocket.readyState === WebSocket.CONNECTING + ) { + // Trying to send now will throw an error. Send it once we get connected. + instance.sendOnceConnected.push(PAYLOAD); + } else { + // CLOSING and CLOSED states won't throw an error, just silently ignore + instance.websocket.send(PAYLOAD); + } + } + + closeWithoutReason(_, utils) { + const instance = this.instances[utils.target.id]; + if (!instance) return; + instance.manuallyClosed = true; + if (instance.websocket) { + instance.websocket.close(); + } + } + + closeWithCode(args, utils) { + const instance = this.instances[utils.target.id]; + if (!instance) return; + instance.manuallyClosed = true; + if (instance.websocket) { + instance.websocket.close(toCloseCode(args.CODE)); + } + } + + closeWithReason(args, utils) { + const instance = this.instances[utils.target.id]; + if (!instance) return; + instance.manuallyClosed = true; + if (instance.websocket) { + instance.websocket.close( + toCloseCode(args.CODE), + toCloseReason(args.REASON) + ); + } + } + } + + // @ts-ignore + Scratch.extensions.register(new WebSocketExtension()); +})(Scratch); diff --git a/images/godslayerakp/ws.png b/images/godslayerakp/ws.png new file mode 100644 index 0000000000..e8fb334daa Binary files /dev/null and b/images/godslayerakp/ws.png differ