From ee61de36f841ae6c90eee6083d69a1f325aae675 Mon Sep 17 00:00:00 2001 From: Logise1123 <88378769+Logise1123@users.noreply.github.com> Date: Mon, 21 Apr 2025 14:55:12 +0200 Subject: [PATCH 1/2] Create firebasedb.js --- extensions/firebasedb.js | 187 +++++++++++++++++++++++++++++++++++++++ 1 file changed, 187 insertions(+) create mode 100644 extensions/firebasedb.js diff --git a/extensions/firebasedb.js b/extensions/firebasedb.js new file mode 100644 index 0000000000..c1805b1cec --- /dev/null +++ b/extensions/firebasedb.js @@ -0,0 +1,187 @@ +// Name: FirebaseDB +// ID: firebasedb +// Description: Extension for online, password-protected Firebase database storage. +// By: Logise +// Original: FirebaseDB +// License: MPL-2.0 + +(function(Scratch) { + 'use strict'; + if (!Scratch.extensions.unsandboxed) throw new Error("FirebaseDB must run unsandboxed"); + const icon = ""; + + class FirebaseDB { + constructor() { + this.api = "https://guessthepin-2fe64-default-rtdb.europe-west1.firebasedatabase.app"; + } + + getInfo() { + return { + id: "FirebaseDB", + name: "Firebase DB", + color1: "#fea631", + menuIconURI: icon, + blocks: [ + {blockType: Scratch.BlockType.LABEL, text: "Made by @logise on Discord"}, + { + opcode: "setKey", + blockType: Scratch.BlockType.COMMAND, + text: "set key [KEY] to value [VALUE]", + arguments: { + KEY: { type: Scratch.ArgumentType.STRING, defaultValue: "key" }, + VALUE: { type: Scratch.ArgumentType.STRING, defaultValue: "value" } + } + }, + { + opcode: "getKey", + blockType: Scratch.BlockType.REPORTER, + text: "get key [KEY]", + arguments: { + KEY: { type: Scratch.ArgumentType.STRING, defaultValue: "key" } + } + }, + {blockType: Scratch.BlockType.LABEL, text: "Password Blocks:"}, + { + opcode: "setKeyWithPassword", + blockType: Scratch.BlockType.COMMAND, + text: "set key [KEY] to value [VALUE] with password [PASSWORD]", + arguments: { + KEY: { type: Scratch.ArgumentType.STRING, defaultValue: "key" }, + VALUE: { type: Scratch.ArgumentType.STRING, defaultValue: "value" }, + PASSWORD: { type: Scratch.ArgumentType.STRING, defaultValue: "password" } + } + }, + { + opcode: "getKeyWithPassword", + blockType: Scratch.BlockType.REPORTER, + text: "get key [KEY] with password [PASSWORD]", + arguments: { + KEY: { type: Scratch.ArgumentType.STRING, defaultValue: "key" }, + PASSWORD: { type: Scratch.ArgumentType.STRING, defaultValue: "password" } + } + }, + { + opcode: "checkPassword", + blockType: Scratch.BlockType.BOOLEAN, + text: "check if password [PASSWORD] is valid for key [KEY]", + arguments: { + PASSWORD: { type: Scratch.ArgumentType.STRING, defaultValue: "password" }, + KEY: { type: Scratch.ArgumentType.STRING, defaultValue: "key" } + } + } + ] + }; + } + + async delay() { + const d = Math.random() * 500; + return new Promise(r => setTimeout(r, d)); + } + + async setKey(args) { + await this.delay(); + const { KEY, VALUE } = args; + if (VALUE.length > 8000) return; + await fetch(`${this.api}/pin/${encodeURIComponent(KEY)}.json`, { + method: "PUT", + body: JSON.stringify(VALUE) + }); + } + + async getKey(args) { + await this.delay(); + const { KEY } = args; + const res = await fetch(`${this.api}/pin/${encodeURIComponent(KEY)}.json`); + const data = await res.json(); + return data ?? ""; + } + + async deriveKey(password, salt) { + const enc = new TextEncoder(); + const keyMaterial = await crypto.subtle.importKey( + "raw", enc.encode(password), "PBKDF2", false, ["deriveKey"] + ); + return await crypto.subtle.deriveKey( + { + name: "PBKDF2", + salt, + iterations: 100000, + hash: "SHA-256" + }, + keyMaterial, + { name: "AES-GCM", length: 256 }, + false, + ["encrypt", "decrypt"] + ); + } + + async setKeyWithPassword(args) { + await this.delay(); + const { KEY, VALUE, PASSWORD } = args; + if (VALUE.length > 8000) return; + + const enc = new TextEncoder(); + const iv = crypto.getRandomValues(new Uint8Array(12)); + const salt = crypto.getRandomValues(new Uint8Array(16)); + const key = await this.deriveKey(PASSWORD, salt); + const encrypted = await crypto.subtle.encrypt( + { name: "AES-GCM", iv }, key, enc.encode(VALUE) + ); + + const fullPackage = { + iv: Array.from(iv), + salt: Array.from(salt), + data: Array.from(new Uint8Array(encrypted)) + }; + + await fetch(`${this.api}/cypher/${encodeURIComponent(KEY)}.json`, { + method: "PUT", + body: JSON.stringify(fullPackage) + }); + } + + async getKeyWithPassword(args) { + await this.delay(); + const { KEY, PASSWORD } = args; + + const res = await fetch(`${this.api}/cypher/${encodeURIComponent(KEY)}.json`); + const encryptedPackage = await res.json(); + if (!encryptedPackage || !encryptedPackage.data || !encryptedPackage.iv || !encryptedPackage.salt) return ""; + + try { + const iv = new Uint8Array(encryptedPackage.iv); + const salt = new Uint8Array(encryptedPackage.salt); + const data = new Uint8Array(encryptedPackage.data); + const key = await this.deriveKey(PASSWORD, salt); + const decrypted = await crypto.subtle.decrypt( + { name: "AES-GCM", iv }, key, data + ); + return new TextDecoder().decode(decrypted); + } catch (e) { + return ""; + } + } + + async checkPassword(args) { + await this.delay(); + const { KEY, PASSWORD } = args; + + const res = await fetch(`${this.api}/cypher/${encodeURIComponent(KEY)}.json`); + const encryptedPackage = await res.json(); + if (!encryptedPackage || !encryptedPackage.data || !encryptedPackage.iv || !encryptedPackage.salt) return false; + + try { + const iv = new Uint8Array(encryptedPackage.iv); + const salt = new Uint8Array(encryptedPackage.salt); + const data = new Uint8Array(encryptedPackage.data); + const key = await this.deriveKey(PASSWORD, salt); + await crypto.subtle.decrypt({ name: "AES-GCM", iv }, key, data); + return true; + } catch (e) { + return false; + } + } + } + + Scratch.extensions.register(new FirebaseDB()); +})(Scratch); From 811b05937dc22c1cb93c41af86acfa96ab267181 Mon Sep 17 00:00:00 2001 From: "DangoCat[bot]" Date: Mon, 21 Apr 2025 14:04:56 +0000 Subject: [PATCH 2/2] [Automated] Format code --- extensions/firebasedb.js | 401 ++++++++++++++++++++++----------------- 1 file changed, 224 insertions(+), 177 deletions(-) diff --git a/extensions/firebasedb.js b/extensions/firebasedb.js index c1805b1cec..29522b3af2 100644 --- a/extensions/firebasedb.js +++ b/extensions/firebasedb.js @@ -5,183 +5,230 @@ // Original: FirebaseDB // License: MPL-2.0 -(function(Scratch) { - 'use strict'; - if (!Scratch.extensions.unsandboxed) throw new Error("FirebaseDB must run unsandboxed"); - const icon = ""; - - class FirebaseDB { - constructor() { - this.api = "https://guessthepin-2fe64-default-rtdb.europe-west1.firebasedatabase.app"; - } - - getInfo() { - return { - id: "FirebaseDB", - name: "Firebase DB", - color1: "#fea631", - menuIconURI: icon, - blocks: [ - {blockType: Scratch.BlockType.LABEL, text: "Made by @logise on Discord"}, - { - opcode: "setKey", - blockType: Scratch.BlockType.COMMAND, - text: "set key [KEY] to value [VALUE]", - arguments: { - KEY: { type: Scratch.ArgumentType.STRING, defaultValue: "key" }, - VALUE: { type: Scratch.ArgumentType.STRING, defaultValue: "value" } - } - }, - { - opcode: "getKey", - blockType: Scratch.BlockType.REPORTER, - text: "get key [KEY]", - arguments: { - KEY: { type: Scratch.ArgumentType.STRING, defaultValue: "key" } - } - }, - {blockType: Scratch.BlockType.LABEL, text: "Password Blocks:"}, - { - opcode: "setKeyWithPassword", - blockType: Scratch.BlockType.COMMAND, - text: "set key [KEY] to value [VALUE] with password [PASSWORD]", - arguments: { - KEY: { type: Scratch.ArgumentType.STRING, defaultValue: "key" }, - VALUE: { type: Scratch.ArgumentType.STRING, defaultValue: "value" }, - PASSWORD: { type: Scratch.ArgumentType.STRING, defaultValue: "password" } - } - }, - { - opcode: "getKeyWithPassword", - blockType: Scratch.BlockType.REPORTER, - text: "get key [KEY] with password [PASSWORD]", - arguments: { - KEY: { type: Scratch.ArgumentType.STRING, defaultValue: "key" }, - PASSWORD: { type: Scratch.ArgumentType.STRING, defaultValue: "password" } - } - }, - { - opcode: "checkPassword", - blockType: Scratch.BlockType.BOOLEAN, - text: "check if password [PASSWORD] is valid for key [KEY]", - arguments: { - PASSWORD: { type: Scratch.ArgumentType.STRING, defaultValue: "password" }, - KEY: { type: Scratch.ArgumentType.STRING, defaultValue: "key" } - } - } - ] - }; - } - - async delay() { - const d = Math.random() * 500; - return new Promise(r => setTimeout(r, d)); - } - - async setKey(args) { - await this.delay(); - const { KEY, VALUE } = args; - if (VALUE.length > 8000) return; - await fetch(`${this.api}/pin/${encodeURIComponent(KEY)}.json`, { - method: "PUT", - body: JSON.stringify(VALUE) - }); - } - - async getKey(args) { - await this.delay(); - const { KEY } = args; - const res = await fetch(`${this.api}/pin/${encodeURIComponent(KEY)}.json`); - const data = await res.json(); - return data ?? ""; - } - - async deriveKey(password, salt) { - const enc = new TextEncoder(); - const keyMaterial = await crypto.subtle.importKey( - "raw", enc.encode(password), "PBKDF2", false, ["deriveKey"] - ); - return await crypto.subtle.deriveKey( - { - name: "PBKDF2", - salt, - iterations: 100000, - hash: "SHA-256" - }, - keyMaterial, - { name: "AES-GCM", length: 256 }, - false, - ["encrypt", "decrypt"] - ); - } - - async setKeyWithPassword(args) { - await this.delay(); - const { KEY, VALUE, PASSWORD } = args; - if (VALUE.length > 8000) return; - - const enc = new TextEncoder(); - const iv = crypto.getRandomValues(new Uint8Array(12)); - const salt = crypto.getRandomValues(new Uint8Array(16)); - const key = await this.deriveKey(PASSWORD, salt); - const encrypted = await crypto.subtle.encrypt( - { name: "AES-GCM", iv }, key, enc.encode(VALUE) - ); - - const fullPackage = { - iv: Array.from(iv), - salt: Array.from(salt), - data: Array.from(new Uint8Array(encrypted)) - }; - - await fetch(`${this.api}/cypher/${encodeURIComponent(KEY)}.json`, { - method: "PUT", - body: JSON.stringify(fullPackage) - }); - } - - async getKeyWithPassword(args) { - await this.delay(); - const { KEY, PASSWORD } = args; - - const res = await fetch(`${this.api}/cypher/${encodeURIComponent(KEY)}.json`); - const encryptedPackage = await res.json(); - if (!encryptedPackage || !encryptedPackage.data || !encryptedPackage.iv || !encryptedPackage.salt) return ""; - - try { - const iv = new Uint8Array(encryptedPackage.iv); - const salt = new Uint8Array(encryptedPackage.salt); - const data = new Uint8Array(encryptedPackage.data); - const key = await this.deriveKey(PASSWORD, salt); - const decrypted = await crypto.subtle.decrypt( - { name: "AES-GCM", iv }, key, data - ); - return new TextDecoder().decode(decrypted); - } catch (e) { - return ""; - } - } - - async checkPassword(args) { - await this.delay(); - const { KEY, PASSWORD } = args; - - const res = await fetch(`${this.api}/cypher/${encodeURIComponent(KEY)}.json`); - const encryptedPackage = await res.json(); - if (!encryptedPackage || !encryptedPackage.data || !encryptedPackage.iv || !encryptedPackage.salt) return false; - - try { - const iv = new Uint8Array(encryptedPackage.iv); - const salt = new Uint8Array(encryptedPackage.salt); - const data = new Uint8Array(encryptedPackage.data); - const key = await this.deriveKey(PASSWORD, salt); - await crypto.subtle.decrypt({ name: "AES-GCM", iv }, key, data); - return true; - } catch (e) { - return false; - } - } +(function (Scratch) { + "use strict"; + if (!Scratch.extensions.unsandboxed) + throw new Error("FirebaseDB must run unsandboxed"); + const icon = + ""; + + class FirebaseDB { + constructor() { + this.api = + "https://guessthepin-2fe64-default-rtdb.europe-west1.firebasedatabase.app"; } - Scratch.extensions.register(new FirebaseDB()); + getInfo() { + return { + id: "FirebaseDB", + name: "Firebase DB", + color1: "#fea631", + menuIconURI: icon, + blocks: [ + { + blockType: Scratch.BlockType.LABEL, + text: "Made by @logise on Discord", + }, + { + opcode: "setKey", + blockType: Scratch.BlockType.COMMAND, + text: "set key [KEY] to value [VALUE]", + arguments: { + KEY: { type: Scratch.ArgumentType.STRING, defaultValue: "key" }, + VALUE: { + type: Scratch.ArgumentType.STRING, + defaultValue: "value", + }, + }, + }, + { + opcode: "getKey", + blockType: Scratch.BlockType.REPORTER, + text: "get key [KEY]", + arguments: { + KEY: { type: Scratch.ArgumentType.STRING, defaultValue: "key" }, + }, + }, + { blockType: Scratch.BlockType.LABEL, text: "Password Blocks:" }, + { + opcode: "setKeyWithPassword", + blockType: Scratch.BlockType.COMMAND, + text: "set key [KEY] to value [VALUE] with password [PASSWORD]", + arguments: { + KEY: { type: Scratch.ArgumentType.STRING, defaultValue: "key" }, + VALUE: { + type: Scratch.ArgumentType.STRING, + defaultValue: "value", + }, + PASSWORD: { + type: Scratch.ArgumentType.STRING, + defaultValue: "password", + }, + }, + }, + { + opcode: "getKeyWithPassword", + blockType: Scratch.BlockType.REPORTER, + text: "get key [KEY] with password [PASSWORD]", + arguments: { + KEY: { type: Scratch.ArgumentType.STRING, defaultValue: "key" }, + PASSWORD: { + type: Scratch.ArgumentType.STRING, + defaultValue: "password", + }, + }, + }, + { + opcode: "checkPassword", + blockType: Scratch.BlockType.BOOLEAN, + text: "check if password [PASSWORD] is valid for key [KEY]", + arguments: { + PASSWORD: { + type: Scratch.ArgumentType.STRING, + defaultValue: "password", + }, + KEY: { type: Scratch.ArgumentType.STRING, defaultValue: "key" }, + }, + }, + ], + }; + } + + async delay() { + const d = Math.random() * 500; + return new Promise((r) => setTimeout(r, d)); + } + + async setKey(args) { + await this.delay(); + const { KEY, VALUE } = args; + if (VALUE.length > 8000) return; + await fetch(`${this.api}/pin/${encodeURIComponent(KEY)}.json`, { + method: "PUT", + body: JSON.stringify(VALUE), + }); + } + + async getKey(args) { + await this.delay(); + const { KEY } = args; + const res = await fetch( + `${this.api}/pin/${encodeURIComponent(KEY)}.json` + ); + const data = await res.json(); + return data ?? ""; + } + + async deriveKey(password, salt) { + const enc = new TextEncoder(); + const keyMaterial = await crypto.subtle.importKey( + "raw", + enc.encode(password), + "PBKDF2", + false, + ["deriveKey"] + ); + return await crypto.subtle.deriveKey( + { + name: "PBKDF2", + salt, + iterations: 100000, + hash: "SHA-256", + }, + keyMaterial, + { name: "AES-GCM", length: 256 }, + false, + ["encrypt", "decrypt"] + ); + } + + async setKeyWithPassword(args) { + await this.delay(); + const { KEY, VALUE, PASSWORD } = args; + if (VALUE.length > 8000) return; + + const enc = new TextEncoder(); + const iv = crypto.getRandomValues(new Uint8Array(12)); + const salt = crypto.getRandomValues(new Uint8Array(16)); + const key = await this.deriveKey(PASSWORD, salt); + const encrypted = await crypto.subtle.encrypt( + { name: "AES-GCM", iv }, + key, + enc.encode(VALUE) + ); + + const fullPackage = { + iv: Array.from(iv), + salt: Array.from(salt), + data: Array.from(new Uint8Array(encrypted)), + }; + + await fetch(`${this.api}/cypher/${encodeURIComponent(KEY)}.json`, { + method: "PUT", + body: JSON.stringify(fullPackage), + }); + } + + async getKeyWithPassword(args) { + await this.delay(); + const { KEY, PASSWORD } = args; + + const res = await fetch( + `${this.api}/cypher/${encodeURIComponent(KEY)}.json` + ); + const encryptedPackage = await res.json(); + if ( + !encryptedPackage || + !encryptedPackage.data || + !encryptedPackage.iv || + !encryptedPackage.salt + ) + return ""; + + try { + const iv = new Uint8Array(encryptedPackage.iv); + const salt = new Uint8Array(encryptedPackage.salt); + const data = new Uint8Array(encryptedPackage.data); + const key = await this.deriveKey(PASSWORD, salt); + const decrypted = await crypto.subtle.decrypt( + { name: "AES-GCM", iv }, + key, + data + ); + return new TextDecoder().decode(decrypted); + } catch (e) { + return ""; + } + } + + async checkPassword(args) { + await this.delay(); + const { KEY, PASSWORD } = args; + + const res = await fetch( + `${this.api}/cypher/${encodeURIComponent(KEY)}.json` + ); + const encryptedPackage = await res.json(); + if ( + !encryptedPackage || + !encryptedPackage.data || + !encryptedPackage.iv || + !encryptedPackage.salt + ) + return false; + + try { + const iv = new Uint8Array(encryptedPackage.iv); + const salt = new Uint8Array(encryptedPackage.salt); + const data = new Uint8Array(encryptedPackage.data); + const key = await this.deriveKey(PASSWORD, salt); + await crypto.subtle.decrypt({ name: "AES-GCM", iv }, key, data); + return true; + } catch (e) { + return false; + } + } + } + + Scratch.extensions.register(new FirebaseDB()); })(Scratch);